home *** CD-ROM | disk | FTP | other *** search
/ CU Amiga Super CD-ROM 15 / CU Amiga Magazine's Super CD-ROM 15 (1997)(EMAP Images)(GB)[!][issue 1997-10].iso / CUCD / Graphics / Ghostscript / source / gdevcdj.c < prev    next >
C/C++ Source or Header  |  1997-03-18  |  115KB  |  3,917 lines

  1. /* Copyright (C) 1991, 1995, 1996, 1997 Aladdin Enterprises.  All rights reserved.
  2.   
  3.   This file is part of Aladdin Ghostscript.
  4.   
  5.   Aladdin Ghostscript is distributed with NO WARRANTY OF ANY KIND.  No author
  6.   or distributor accepts any responsibility for the consequences of using it,
  7.   or for whether it serves any particular purpose or works at all, unless he
  8.   or she says so in writing.  Refer to the Aladdin Ghostscript Free Public
  9.   License (the "License") for full details.
  10.   
  11.   Every copy of Aladdin Ghostscript must include a copy of the License,
  12.   normally in a plain ASCII text file named PUBLIC.  The License grants you
  13.   the right to copy, modify and redistribute Aladdin Ghostscript, but only
  14.   under certain conditions described in the License.  Among other things, the
  15.   License requires that the copyright notice and this notice be preserved on
  16.   all copies.
  17. */
  18.  
  19. /* gdevcdj.c */
  20. /* H-P and Canon colour printer drivers */
  21.  
  22. /****************************************************************
  23.  * The code in this file has gotten completely out of hand.
  24.  * Too many users have made too many "improvements" without
  25.  * regard to the overall structure; the last three "improvements"
  26.  * required me to spend several hours fixing each one so that
  27.  * the code worked again at all.  For this reason, no further
  28.  * changes to this file will be accepted.  I am planning eventually
  29.  * to get these drivers rewritten from scratch.
  30.  *                L. Peter Deutsch
  31.  *                Aladdin Enterprises
  32.  *                February 28, 1996
  33.  ****************************************************************/
  34.  
  35. /*
  36.  * Important compilation notes (YA).
  37.  *
  38.  * You may also try the cdj550cmyk driver after having defined
  39.  * USE_CDJ550_CMYK and added the needed definition in devs.mak. Not tried!
  40.  * (I have a BJC!) If you ty that, please report success/failure to me,
  41.  * Yves.Arrouye@imag.fr. Also note that modes descriptions of CMYK printing
  42.  * is done under the BJC section of devices.doc.
  43.  *
  44.  * CMYK to RGB conversion is made a la GhostScript unless you define
  45.  * the preprocessor symbol USE_ADOBE_CMYK_RGB.
  46.  *
  47.  *   Ghostscript:    R = (1.0 - C) * (1.0 - K)
  48.  *   Adobe:        R = 1.0 - min(1.0, C + K)
  49.  *
  50.  * (and similarly for G and B).  Ghostscript claims its method achieves
  51.  * better results.
  52.  *
  53.  * For the BJC drivers, define BJC_DEFAULT_CENTEREDAREA if you want to
  54.  * have the  same top and bottom margins (default to use the tallest
  55.  * imageable area available, usually with a top margin smaller than
  56.  * the bottom one). Defining USE_RECOMMENDED_MARGINS has the same
  57.  * effect and also sets these margins to 12.4 mm. Other compilation
  58.  * defines are explained in devices.doc.
  59.  *
  60.  * You can also define BJC_INIT_800_AS_600 to not use BJC-800-specific code
  61.  * in the page initialization sequence (normally not useful to you at all,
  62.  * just for my debugging of the driver margins).
  63.  *
  64.  */
  65.  
  66. #include "std.h"        /* to stop stdlib.h redefining types */
  67. #include <stdlib.h>        /* for rand() */
  68. #include "gdevprn.h"
  69. #include "gdevpcl.h"
  70. #include "gsparam.h"
  71. #include "gsstate.h"
  72.  
  73. /* Conversion stuff. */
  74. #include "gxlum.h"
  75.  
  76. /* Canon stuff */
  77.  
  78. #include "gdevbjc.h"
  79.  
  80. /***
  81.  *** This file contains multiple drivers.  The main body of code, and all
  82.  *** but the DesignJet driver, were contributed by George Cameron;
  83.  *** please contact g.cameron@biomed.abdn.ac.uk if you have questions.
  84.  *     1 - cdj500:    HP DeskJet 500C
  85.  *     2 - cdj550:    HP DeskJet 550C
  86.  *     3 - pjxl300:    HP PaintJet XL300
  87.  *     4 - pj:        HP PaintJet
  88.  *     5 - pjxl:    HP PaintJet XL
  89.  *     6 - declj250:    DEC LJ250
  90.  *** The DesignJet 650C driver was contributed by Koert Zeilstra;
  91.  *** please contact koert@zen.cais.com if you have questions.
  92.  *     7 - dnj650c      HP DesignJet 650C
  93.  *** The LaserJet 4 driver with dithering was contributed by Eckhard
  94.  *** Rueggeberg; please contact eckhard@ts.go.dlr.de if you have questions.
  95.  *     8 - lj4dith:    HP LaserJet 4 with dithering
  96.  *** The ESC/P driver (for Epson ESC/P compatible printers) was written by
  97.  *** Yoshio Kuniyoshi <yoshio@nak.math.keio.ac.jp>, but is not maintained at
  98.  *** the moment.
  99.  *     9 - esc/p:       Epson ESC/P-compatible printers
  100.  *** The BJC600 driver (which also works for BJC4000) was written first
  101.  *** by Yoshio Kuniyoshi <yoshio@nak.math.keio.ac.jp> and later modified by
  102.  *** Yves Arrouye <Yves.Arrouye@imag.fr>. The current driver has been
  103.  *** completely rewritten by me (YA) for good color handling.
  104.  *    10 - bjc600:    BJC 600//4000 printers
  105.  *** The BJC800 driver is based on the bjc600 one. By YA too.
  106.  *    11 - bjc800:    BJC 800 printer
  107.  ***/
  108.  
  109. /*
  110.  * All of the HP-like drivers have 8-bit (monochrome), 16-bit and 24-bit
  111.  *     (colour) and for the DJ 550C 32-bit, (colour, cmyk mode)
  112.  *     options in addition to the usual 1-bit and 3-bit modes
  113.  * It is also possible to set various printer-specific parameters
  114.  *     from the gs command line, eg.
  115.  *
  116.  *  gs -sDEVICE=cdj550 -dBitsPerPixel=16 -dDepletion=1 -dShingling=2 tiger.ps
  117.  *
  118.  * Please consult the appropriate section in the devices.doc file for
  119.  * further details on all these drivers.
  120.  *
  121.  * All of the BJC-like drivers have 1-bit and 8-bit monochrome modes, 8-bit,
  122.  *     16-bit, 24-bit and 32-bit colour cmyk mode (the 8-bit monochrome mode
  123.  *     is called "4-bit". If you want to add a CMYK printer, look at the
  124.  *     bjc6000/bjc800 devices declarations and initialization.
  125.  *
  126.  * If you want to support different color components for the same depth
  127.  * on a non-CMYK printer, look how this is done for CMYK printers in
  128.  * cdj_set_bpp.
  129.  *
  130.  */
  131.  
  132. /*
  133.  * This taken from gsdparam.c. I hope it will be useable directly some day.
  134.  *
  135.  */
  136.  
  137. #define BEGIN_ARRAY_PARAM(pread, pname, pa, psize, e)\
  138.   switch ( ncode = pread(plist, (oname = pname), &pa) )\
  139.   {\
  140.   case 0:\
  141.     if ( pa.size != psize )\
  142.       code = gs_error_rangecheck;\
  143.     else { 
  144. /* The body of the processing code goes here. */
  145. /* If it succeeds, it should do a 'break'; */
  146. /* if it fails, it should set ecode and fall through. */
  147. #define END_PARAM(pa, e)\
  148.     }\
  149.     goto e;\
  150.   default:\
  151.     code = ncode;\
  152. e:    param_signal_error(plist, oname, code);\
  153.   case 1:\
  154.     pa.data = 0;        /* mark as not filled */\
  155.   }
  156.  
  157. private int cdj_param_check_bytes(P5(gs_param_list *, gs_param_name, const byte *, uint, bool));
  158. private int cdj_param_check_float(P4(gs_param_list *, gs_param_name, floatp, bool));
  159. #define cdj_param_check_string(plist, pname, str, defined)\
  160.   cdj_param_check_bytes(plist, pname, (const byte *)str, strlen(str), defined)
  161.  
  162. /*
  163.  * Drivers stuff.
  164.  *
  165.  */
  166.  
  167. #define DESKJET_PRINT_LIMIT  0.04    /* 'real' top margin? */
  168. #define PAINTJET_PRINT_LIMIT 0.0    /* This is a guess.. */
  169. #define ESC_P_PRINT_LIMIT    0.335
  170.  
  171. /* Margins are left, bottom, right, top. */
  172. #define DESKJET_MARGINS_LETTER   0.25, 0.50, 0.25, 0.167
  173. #define DESKJET_MARGINS_A4       0.125, 0.50, 0.143, 0.167
  174. #define LJET4_MARGINS           0.26, 0.0, 0.0, 0.0
  175. /* The PaintJet and DesignJet seem to have the same margins */
  176. /* regardless of paper size. */
  177. #define PAINTJET_MARGINS         0.167, 0.167, 0.167, 0.167
  178. #define DESIGNJET_MARGINS        0.167, 0.167, 0.167, 0.167
  179.  
  180. /*
  181.  * With ESC/P commands, BJC-600 can print no more than 8 inches width.
  182.  * So it can't use full width of letter size paper.  Since non printable
  183.  * left side area is 0.134 inch, we set up margins as follows.
  184.  *
  185.  * Note to readers: the bjc drivers series do *not* use ESC/P commands
  186.  * but raster ops. Configuration of these drivers can be done through
  187.  * the gdevbjc.h file.
  188.  *
  189.  */
  190. #define ESC_P_MARGINS_LETTER    0.134, 0.276+0.2, 0.366+0.01, 0.335
  191. #define ESC_P_MARGINS_A4        0.134, 0.276+0.2, 0.166+0.01, 0.335
  192.  
  193. /* Define bits-per-pixel for generic drivers - default is 24-bit mode */
  194. #ifndef BITSPERPIXEL
  195. #  define BITSPERPIXEL 24
  196. #endif
  197.  
  198. #define W sizeof(word)
  199. #define I sizeof(int)
  200.  
  201.  
  202. #define invert_word(v)\
  203.      ((v) >> 24) + (((v) >> 8) & 0xff00L) +\
  204.      (((word)(v) << 8) & 0xff0000L) + ((word)(v) << 24)
  205.  
  206.  
  207. /* Printer types */
  208. #define DJ500C   0
  209. #define DJ550C   1
  210. #define PJXL300  2
  211. #define PJ180    3
  212. #define PJXL180  4
  213. #define DECLJ250 5
  214. #define DNJ650C  6
  215. #define LJ4DITH  7
  216. #define ESC_P     8
  217. #define BJC600     9
  218. #define BJC800     10
  219.  
  220. /* No. of ink jets (used to minimise head movements) */
  221. #define HEAD_ROWS_MONO 50
  222. #define HEAD_ROWS_COLOUR 16
  223.  
  224. /* Colour mapping procedures */
  225. private dev_proc_map_cmyk_color (gdev_cmyk_map_cmyk_color);
  226. private dev_proc_map_rgb_color (gdev_cmyk_map_rgb_color);
  227. private dev_proc_map_color_rgb (gdev_cmyk_map_color_rgb);
  228.  
  229. private dev_proc_map_rgb_color (gdev_pcl_map_rgb_color);
  230. private dev_proc_map_color_rgb (gdev_pcl_map_color_rgb);
  231.  
  232. /* Print-page, parameters and miscellaneous procedures */
  233. private dev_proc_open_device(dj500c_open);
  234. private dev_proc_open_device(dj550c_open);
  235. private dev_proc_open_device(dnj650c_open);
  236. private dev_proc_open_device(lj4dith_open);
  237. private dev_proc_open_device(pj_open);
  238. private dev_proc_open_device(pjxl_open);
  239. private dev_proc_open_device(pjxl300_open);
  240. private dev_proc_open_device(escp_open);
  241. private dev_proc_open_device(bjc_open);
  242.  
  243. private dev_proc_print_page(declj250_print_page);
  244. private dev_proc_print_page(dj500c_print_page);
  245. private dev_proc_print_page(dj550c_print_page);
  246. private dev_proc_print_page(dnj650c_print_page);
  247. private dev_proc_print_page(lj4dith_print_page);
  248. private dev_proc_print_page(pj_print_page);
  249. private dev_proc_print_page(pjxl_print_page);
  250. private dev_proc_print_page(pjxl300_print_page);
  251. private dev_proc_print_page(escp_print_page);
  252. private dev_proc_print_page(bjc_print_page);
  253.  
  254. private dev_proc_get_params(cdj_get_params);
  255. private dev_proc_get_params(pjxl_get_params);
  256. private dev_proc_get_params(bjc_get_params);
  257. #define ep_get_params cdj_get_params
  258.  
  259. private dev_proc_put_params(cdj_put_params);
  260. private dev_proc_put_params(pj_put_params);
  261. private dev_proc_put_params(pjxl_put_params);
  262. private dev_proc_put_params(bjc_put_params);
  263. #define ep_put_params cdj_put_params
  264.  
  265. /* The device descriptors */
  266.  
  267. #define gx_prn_colour_device_common \
  268.     gx_prn_device_common; \
  269.     short cmyk;          /* 0: not CMYK-capable, > 0: printing CMYK, */ \
  270.               /* < 0 : CMYK-capable, not printing CMYK */ \
  271.     uint default_depth;    /* Used only for CMYK-capable printers now. */ \
  272.     uint correction
  273.  
  274. typedef struct gx_device_cdj_s gx_device_cdj;
  275. struct gx_device_cdj_s {
  276.     gx_device_common;
  277.     gx_prn_colour_device_common;
  278.     int shingling;          /* Interlaced, multi-pass printing */
  279.     int depletion;          /* 'Intelligent' dot-removal */
  280. };
  281.  
  282. typedef struct gx_device_pjxl_s gx_device_pjxl;
  283. struct gx_device_pjxl_s {
  284.     gx_device_common;
  285.     gx_prn_colour_device_common;
  286.     int printqual;            /* Mechanical print quality */
  287.     int rendertype;           /* Driver or printer dithering control */
  288. };
  289.  
  290. typedef struct gx_device_hp_s gx_device_hp;
  291. struct gx_device_hp_s {
  292.     gx_device_common;
  293.     gx_prn_colour_device_common;
  294. };
  295.  
  296. typedef struct gx_device_hp_s gx_device_pj;
  297.  
  298. typedef struct gx_device_bjc600_s gx_device_bjc600;
  299. typedef struct gx_device_bjc800_s gx_device_bjc800;
  300.  
  301. typedef struct gx_device_bjc800_s gx_device_bjc;
  302.  
  303. #define bjc_params_common \
  304.     bool manualFeed;        /* Use manual feed */ \
  305.     int mediaType;        /* Cf. strings below */ \
  306.     bool mediaWeight_isSet;    /* Say if weight is an integer or null */ \
  307.     int mediaWeight;        /* Weigth of the media */ \
  308.     int printQuality;        /* Cf. strings below */ \
  309.     bool ditheringType;        /* Do dithering */ \
  310.     int colorComponents;    /* The number of *desired* color comps */ \
  311.     int printColors        /* 0: Transparent, \
  312.                    1: C, 2: M, 4: Y, 7: K (Color decomp). \
  313.                    if > 8, print in black ink. */
  314.  
  315. typedef struct {
  316.     bjc_params_common;
  317.  
  318.     bool monochromePrint;    /* Print with black only */
  319. } bjc600_params;
  320.  
  321. typedef struct {
  322.     bjc_params_common;
  323. } bjc_params;
  324.  
  325. typedef bjc_params bjc800_params;
  326.  
  327. #define gx_bjc_device_common \
  328.     gx_device_common; \
  329.     gx_prn_colour_device_common; \
  330.     int ptype; \
  331.     float printLimit
  332.  
  333. struct gx_device_bjc600_s {
  334.         gx_bjc_device_common;
  335.     bjc600_params bjc_p;
  336. };
  337. struct gx_device_bjc800_s {
  338.         gx_bjc_device_common;
  339.     bjc800_params bjc_p;
  340. };
  341.  
  342. typedef struct {
  343.     gx_device_common;
  344.     gx_prn_colour_device_common;
  345. } gx_device_colour_prn;
  346.  
  347. /* Use the cprn_device macro to access generic fields (like cmyk,
  348.    default_depth and correction), and specific macros for specific
  349.    devices. */
  350.  
  351. #define cprn_device     ((gx_device_colour_prn*) pdev)
  352.  
  353. #define cdj       ((gx_device_cdj *)pdev)
  354. #define pjxl      ((gx_device_pjxl *)pdev)
  355. #define pj    ((gx_device_pj *)pdev)
  356.  
  357. #define bjc             ((gx_device_bjc*) pdev)
  358. #define bjc600          ((gx_device_bjc600*) pdev)
  359. #define bjc800          ((gx_device_bjc800*) pdev)
  360.  
  361. #define bjcparams    (bjc->bjc_p)
  362. #define bjc600params    (bjc600->bjc_p)
  363. #define bjc800params    (bjc800->bjc_p)
  364.  
  365. #define bjcversion(p)    (((gx_device_bjc*) pdev)->ptype == BJC800 ? \
  366.     BJC_BJC800_VERSION : BJC_BJC600_VERSION)
  367. #define bjcversionstring(p)    (((gx_device_bjc*) pdev)->ptype == BJC800 ? \
  368.     BJC_BJC800_VERSIONSTR : BJC_BJC600_VERSIONSTR)
  369.  
  370. #define bjcthickpaper(l) \
  371.     (bjcparams.mediaWeight_isSet && bjcparams.mediaWeight > l)
  372. #define bjc600thickpaper() bjcthickpaper(BJC600_MEDIAWEIGHT_THICKLIMIT)
  373. #define bjc800thickpaper() bjcthickpaper(BJC800_MEDIAWEIGHT_THICKLIMIT)
  374.  
  375. /* The basic structure for all printers. Note the presence of the cmyk, depth
  376.    and correct fields even if soem are not used by all printers. */
  377.  
  378. #define prn_colour_device_body(dtype, procs, dname, w10, h10, xdpi, ydpi, lm, bm, rm, tm, ncomp, depth, mg, mc, dg, dc, print_page, cmyk, correct)\
  379.     prn_device_body(dtype, procs, dname, w10, h10, xdpi, ydpi, lm, bm, rm, tm, ncomp, depth, mg, mc, dg, dc, print_page), cmyk, depth /* default */, correct
  380.  
  381. /* Note: the computation of color_info values here must match */
  382. /* the computation in the cdj_set_bpp procedure below. */
  383.  
  384. #define prn_hp_colour_device(dtype, procs, dev_name, x_dpi, y_dpi, bpp, print_page, correct)\
  385.     prn_colour_device_body(dtype, procs, dev_name,\
  386.     DEFAULT_WIDTH_10THS, DEFAULT_HEIGHT_10THS, x_dpi, y_dpi, 0, 0, 0, 0,\
  387.     (bpp == 32 ? 4 : (bpp == 1 || bpp == 8) ? 1 : 3), bpp,\
  388.     (bpp >= 8 ? 255 : 1), (bpp >= 8 ? 255 : bpp > 1 ? 1 : 0),\
  389.     (bpp >= 8 ? 5 : 2), (bpp >= 8 ? 5 : bpp > 1 ? 2 : 0),\
  390.     print_page, 0 /* cmyk */, correct)
  391.  
  392. #define prn_cmyk_colour_device(dtype, procs, dev_name, x_dpi, y_dpi, bpp, print_page, correct)\
  393.     prn_colour_device_body(dtype, procs, dev_name,\
  394.     DEFAULT_WIDTH_10THS, DEFAULT_HEIGHT_10THS, x_dpi, y_dpi, 0, 0, 0, 0,\
  395.     ((bpp == 1 || bpp == 4) ? 1 : 4), bpp,\
  396.     (bpp > 8 ? 255 : 1), (1 << (bpp >> 2)) - 1, /* max_gray, max_color */\
  397.     (bpp > 8 ? 5 : 2), (bpp > 8 ? 5 : bpp > 1 ? 2 : 0),\
  398.     print_page, 1 /* cmyk */, correct)
  399.  
  400. #define bjc_device(dtype, p, d, x, y, b, pp, c) \
  401.     prn_cmyk_colour_device(dtype, p, d, x, y, b, pp, c)
  402.  
  403. #define cdj_device(procs, dev_name, x_dpi, y_dpi, bpp, print_page, correction, shingling, depletion)\
  404. { prn_hp_colour_device(gx_device_cdj, procs, dev_name, x_dpi, y_dpi, bpp, print_page, correction),\
  405.     shingling,\
  406.     depletion\
  407. }
  408.  
  409. #define pjxl_device(procs, dev_name, x_dpi, y_dpi, bpp, print_page, printqual, rendertype)\
  410. { prn_hp_colour_device(gx_device_pjxl, procs, dev_name, x_dpi, y_dpi, bpp, print_page, 0), \
  411.     printqual,\
  412.     rendertype\
  413. }
  414.  
  415. #define pj_device(procs, dev_name, x_dpi, y_dpi, bpp, print_page)\
  416. { prn_hp_colour_device(gx_device_pj, procs, dev_name, x_dpi, y_dpi, bpp, print_page, 0) }
  417.  
  418. #define bjc600_device(procs, dev_name, x_dpi, y_dpi, bpp, print_page, t, mf, mt, mws, mw, pq, dt, cc, pc, mp) \
  419. { bjc_device(gx_device_bjc600, procs, dev_name, x_dpi, y_dpi, bpp, print_page, 0),\
  420.     t, 0., { mf, mt, mws, mw, pq, dt, cc, pc, mp }\
  421. }
  422. #define bjc800_device(procs, dev_name, x_dpi, y_dpi, bpp, print_page, t, mf, mt, mws, mw, pq, dt, cc, pc) \
  423. { bjc_device(gx_device_bjc800, procs, dev_name, x_dpi, y_dpi, bpp, print_page, 0),\
  424.     t, 0., { mf, mt, mws, mw, pq, dt, cc, pc }\
  425. }
  426.  
  427. #define hp_colour_procs(proc_colour_open, proc_get_params, proc_put_params) {\
  428.     proc_colour_open,\
  429.     gx_default_get_initial_matrix,\
  430.     gx_default_sync_output,\
  431.     gdev_prn_output_page,\
  432.     gdev_prn_close,\
  433.     gdev_pcl_map_rgb_color,\
  434.     gdev_pcl_map_color_rgb,\
  435.     NULL,    /* fill_rectangle */\
  436.     NULL,    /* tile_rectangle */\
  437.     NULL,    /* copy_mono */\
  438.     NULL,    /* copy_color */\
  439.     NULL,    /* draw_line */\
  440.     gx_default_get_bits,\
  441.     proc_get_params,\
  442.     proc_put_params\
  443. }
  444.  
  445. #define cmyk_colour_procs(proc_colour_open, proc_get_params, proc_put_params) {\
  446.     proc_colour_open,\
  447.     gx_default_get_initial_matrix,\
  448.     gx_default_sync_output,\
  449.     gdev_prn_output_page,\
  450.     gdev_prn_close,\
  451.     NULL /* map_rgb_color */,\
  452.     gdev_cmyk_map_color_rgb,\
  453.     NULL /* fill_rectangle */,\
  454.     NULL /* tile_rectangle */,\
  455.     NULL /* copy_mono */,\
  456.     NULL /* copy_color */,\
  457.     NULL /* draw_line */,\
  458.     gx_default_get_bits,\
  459.     proc_get_params,\
  460.     proc_put_params,\
  461.         gdev_cmyk_map_cmyk_color\
  462. }
  463.  
  464. private gx_device_procs cdj500_procs =
  465. hp_colour_procs(dj500c_open, cdj_get_params, cdj_put_params);
  466.  
  467. private gx_device_procs cdj550_procs =
  468. hp_colour_procs(dj550c_open, cdj_get_params, cdj_put_params);
  469.  
  470. #ifdef USE_CDJ550_CMYK
  471. private gx_device_procs cdj550cmyk_procs =
  472. cmyk_colour_procs(dj550c_open, cdj_get_params, cdj_put_params);
  473. #endif
  474.  
  475. private gx_device_procs dnj650c_procs =
  476. hp_colour_procs(dnj650c_open, cdj_get_params, cdj_put_params);
  477.  
  478. private gx_device_procs lj4dith_procs =
  479. hp_colour_procs(lj4dith_open, cdj_get_params, cdj_put_params);
  480.  
  481. private gx_device_procs pj_procs =
  482. hp_colour_procs(pj_open, gdev_prn_get_params, pj_put_params);
  483.  
  484. private gx_device_procs pjxl_procs =
  485. hp_colour_procs(pjxl_open, pjxl_get_params, pjxl_put_params);
  486.  
  487. private gx_device_procs pjxl300_procs =
  488. hp_colour_procs(pjxl300_open, pjxl_get_params, pjxl_put_params);
  489.  
  490. private gx_device_procs bjc_procs =
  491. cmyk_colour_procs(bjc_open, bjc_get_params, bjc_put_params);
  492.  
  493. private gx_device_procs escp_procs =
  494. hp_colour_procs(escp_open, ep_get_params, ep_put_params);
  495.  
  496. gx_device_cdj far_data gs_cdjmono_device =
  497. cdj_device(cdj500_procs, "cdjmono", 300, 300, 1,
  498.        dj500c_print_page, 4, 0, 1);
  499.  
  500. gx_device_cdj far_data gs_cdeskjet_device =
  501. cdj_device(cdj500_procs, "cdeskjet", 300, 300, 3,
  502.        dj500c_print_page, 4, 2, 1);
  503.  
  504. gx_device_cdj far_data gs_cdjcolor_device =
  505. cdj_device(cdj500_procs, "cdjcolor", 300, 300, 24,
  506.        dj500c_print_page, 4, 2, 1);
  507.  
  508. gx_device_cdj far_data gs_cdj500_device =
  509. cdj_device(cdj500_procs, "cdj500", 300, 300, BITSPERPIXEL,
  510.        dj500c_print_page, 4, 2, 1);
  511.  
  512. gx_device_cdj far_data gs_cdj550_device =
  513. cdj_device(cdj550_procs, "cdj550", 300, 300, BITSPERPIXEL,
  514.        dj550c_print_page, 0, 2, 1);
  515.  
  516. #ifdef USE_CDJ550_CMYK
  517. gx_device_cdj far_data gs_cdj550cmyk_device = {
  518.     prn_cmyk_colour_device(cdj550cmyk_procs, "cdj550cmyk", 300, 300,
  519.         BITSPERPIXEL, dj550c_print_page, 0), 2, 1
  520. };
  521. #endif
  522.  
  523. gx_device_pj far_data gs_declj250_device =
  524. pj_device(pj_procs, "declj250", 180, 180, BITSPERPIXEL,
  525.       declj250_print_page);
  526.  
  527. gx_device_cdj far_data gs_dnj650c_device =
  528. cdj_device(dnj650c_procs, "dnj650c", 300, 300, BITSPERPIXEL,
  529.        dnj650c_print_page, 0, 2, 1);
  530.        
  531. gx_device_cdj far_data gs_lj4dith_device =
  532. cdj_device(lj4dith_procs, "lj4dith", 600, 600, 8,
  533.        lj4dith_print_page, 4, 0, 1);
  534.  
  535. gx_device_pj far_data gs_pj_device =
  536. pj_device(pj_procs, "pj", 180, 180, BITSPERPIXEL,
  537.       pj_print_page);
  538.  
  539. gx_device_pjxl far_data gs_pjxl_device =
  540. pjxl_device(pjxl_procs, "pjxl", 180, 180, BITSPERPIXEL,
  541.         pjxl_print_page, 0, 0);
  542.  
  543. gx_device_pjxl far_data gs_pjxl300_device =
  544. pjxl_device(pjxl300_procs, "pjxl300", 300, 300, BITSPERPIXEL,
  545.         pjxl300_print_page, 0, 0);
  546.  
  547. gx_device_cdj far_data gs_escp_device =
  548. cdj_device(escp_procs, "escp", 360, 360, 8,
  549.        escp_print_page, 0, 0, 1);
  550.  
  551. gx_device_cdj far_data gs_escpc_device =
  552. cdj_device(escp_procs, "escpc", 360, 360, 24,
  553.        escp_print_page, 0, 0, 1);
  554.  
  555. /* Args of bjc drivers are manualFeed, mediaType, printQuality, printColor,
  556.    mediaWeight_isSet, mediaWeight, (monochromePrint) */
  557.  
  558. gx_device_bjc600 far_data gs_bjc600_device =
  559.     bjc600_device(
  560.     bjc_procs,
  561.         BJC_BJC600,
  562.         BJC600_DEFAULT_RESOLUTION,
  563.         BJC600_DEFAULT_RESOLUTION,
  564.         BJC600_DEFAULT_BITSPERPIXEL,
  565.     bjc_print_page,
  566.         BJC600,
  567.     BJC600_DEFAULT_MANUALFEED,
  568.     BJC600_DEFAULT_MEDIATYPE,
  569.     BJC600_DEFAULT_SETMEDIAWEIGHT,
  570.     BJC600_DEFAULT_MEDIAWEIGHT,
  571.     BJC600_DEFAULT_PRINTQUALITY,
  572.     BJC600_DEFAULT_DITHERINGTYPE,
  573.     BJC600_DEFAULT_COLORCOMPONENTS,
  574.     BJC600_DEFAULT_PRINTCOLORS,
  575.     BJC600_DEFAULT_MONOCHROMEPRINT);
  576.  
  577. gx_device_bjc800 far_data gs_bjc800_device =
  578.     bjc800_device(
  579.         bjc_procs,
  580.         BJC_BJC800,
  581.         BJC800_DEFAULT_RESOLUTION,
  582.         BJC800_DEFAULT_RESOLUTION,
  583.         BJC800_DEFAULT_BITSPERPIXEL,
  584.     bjc_print_page,
  585.         BJC800,
  586.     BJC800_DEFAULT_MANUALFEED,
  587.     BJC800_DEFAULT_MEDIATYPE,
  588.     BJC800_DEFAULT_SETMEDIAWEIGHT,
  589.     BJC800_DEFAULT_MEDIAWEIGHT,
  590.     BJC800_DEFAULT_PRINTQUALITY,
  591.     BJC800_DEFAULT_DITHERINGTYPE,
  592.     BJC600_DEFAULT_COLORCOMPONENTS,
  593.     BJC800_DEFAULT_PRINTCOLORS);
  594.  
  595. /* Forward references */
  596. private int gdev_pcl_mode1compress(P3(const byte *, const byte *, byte *));
  597. private int gdev_pcl_mode9compress(P4(int, const byte *, const byte *, byte *));
  598. private int hp_colour_open(P2(gx_device *, int));
  599. private int hp_colour_print_page(P3(gx_device_printer *, FILE *, int));
  600. private int near cdj_put_param_int(P6(gs_param_list *, gs_param_name, int *, int, int, int));
  601. private uint gdev_prn_rasterwidth(P2(const gx_device_printer *, int));
  602. private int cdj_put_param_bpp(P5(gx_device *, gs_param_list *, int, int, int));
  603. private int cdj_set_bpp(P3(gx_device *, int, int));
  604. private void cdj_expand_line(P5(word *, int, short, int, int));
  605. private int bjc_fscmyk(P5(byte**, byte*[4][4], int**, int, int));
  606.  
  607. /* String parameters manipulation */
  608.  
  609. typedef struct {
  610.     const char* p_name;
  611.     int p_value;
  612. } stringParamDescription;
  613.  
  614. private const byte* paramValueToString(P2(const stringParamDescription*, int));
  615. private int paramStringValue(P4(const stringParamDescription*,
  616.     const byte*, int, int*));
  617.  
  618. private int put_param_string(P6(gs_param_list*, const byte*,
  619.     gs_param_string*, const stringParamDescription*, int *, int));
  620. private int get_param_string(P7(gs_param_list*, const byte*,
  621.     gs_param_string*, const stringParamDescription*, int, int, int));
  622.  
  623. /* Open the printer and set up the margins. */
  624. private int
  625. dj500c_open(gx_device *pdev)
  626. { return hp_colour_open(pdev, DJ500C);
  627. }
  628.  
  629. private int
  630. dj550c_open(gx_device *pdev)
  631. {  return hp_colour_open(pdev, DJ550C);
  632. }
  633.  
  634. private int
  635. dnj650c_open(gx_device *pdev)
  636. { return hp_colour_open(pdev, DNJ650C);
  637. }
  638.  
  639. private int
  640. lj4dith_open(gx_device *pdev)
  641. { return hp_colour_open(pdev, LJ4DITH);
  642. }
  643.  
  644. private int
  645. pjxl300_open(gx_device *pdev)
  646. { return hp_colour_open(pdev, PJXL300);
  647. }
  648.  
  649. private int
  650. pj_open(gx_device *pdev)
  651. { return hp_colour_open(pdev, PJ180);
  652. }
  653.  
  654. private int
  655. pjxl_open(gx_device *pdev)
  656. { return hp_colour_open(pdev, PJXL180);
  657. }
  658.  
  659. private int
  660. escp_open(gx_device *pdev)
  661. { return hp_colour_open(pdev, ESC_P);
  662. }
  663.  
  664. private int
  665. bjc_open(gx_device *pdev)
  666. {  return hp_colour_open(pdev, bjc->ptype);
  667. }
  668.  
  669. private int
  670. hp_colour_open(gx_device *pdev, int ptype)
  671. {    /* Change the margins if necessary. */
  672.   static const float dj_a4[4] = { DESKJET_MARGINS_A4 };
  673.   static const float dj_letter[4] = { DESKJET_MARGINS_LETTER };
  674.   static const float lj4_all[4] = { LJET4_MARGINS };
  675.   static const float pj_all[4] = { PAINTJET_MARGINS };
  676.   static const float dnj_all[4] = { DESIGNJET_MARGINS };
  677.   static const float ep_a4[4] = { ESC_P_MARGINS_A4 };
  678.   static const float ep_letter[4] = { ESC_P_MARGINS_LETTER };
  679.  
  680.   static float bjc_a3[4] = { BJC_MARGINS_A3 };        /* Not const! */
  681.   static float bjc_letter[4] = { BJC_MARGINS_LETTER };  /* Not const! */
  682.   static float bjc_a4[4] = { BJC_MARGINS_A4 };        /* Not const! */
  683.  
  684.   const float _ds *m = (float _ds*) 0;
  685.  
  686.   /* Set up colour params if put_params has not already done so */
  687.   if (pdev->color_info.num_components == 0)
  688.     {    int code = cdj_set_bpp(pdev, pdev->color_info.depth,
  689.         pdev->color_info.num_components);
  690.     if ( code < 0 )
  691.       return code;
  692.     }
  693.  
  694.   switch (ptype) {
  695.   case DJ500C:
  696.   case DJ550C:
  697.     m = (gdev_pcl_paper_size(pdev) == PAPER_SIZE_A4 ? dj_a4 :
  698.      dj_letter);
  699.     break;
  700.   case DNJ650C:
  701.     m = dnj_all;
  702.     break;
  703.   case LJ4DITH:
  704.     m = lj4_all;
  705.     break;  
  706.   case PJ180:
  707.   case PJXL300:
  708.   case PJXL180:
  709.     m = pj_all;
  710.     break;
  711.   case ESC_P:
  712.     m = (gdev_pcl_paper_size(pdev) == PAPER_SIZE_A4 ? ep_a4 :
  713.      ep_letter);
  714.     break;
  715.   case BJC600:
  716.   case BJC800:
  717.     switch (gdev_pcl_paper_size(pdev)) {
  718.     case PAPER_SIZE_LEGAL:
  719.     case PAPER_SIZE_LETTER:
  720.         m = bjc_letter;
  721.         break;
  722.  
  723.     case PAPER_SIZE_A0:
  724.     case PAPER_SIZE_A1:
  725.     case PAPER_SIZE_A3:
  726.         m = bjc_a3;
  727.         break;
  728.  
  729.     default:
  730.         m = bjc_a4;
  731.     }
  732.  
  733. #ifndef USE_FIXED_MARGINS
  734.     if (ptype == BJC800) {
  735.     ((float _ds*) m)[1] = BJC_HARD_LOWER_LIMIT;
  736.     }
  737. #endif
  738.  
  739.     bjc->printLimit = m[3];        /* The real hardware limit. */
  740.  
  741. #ifdef BJC_DEFAULT_CENTEREDAREA
  742.     if (m[3] < m[1]) {
  743.     ((float _ds*) m)[3] = m[1];      /* Top margin = bottom one. */
  744.     } else {
  745.     ((float _ds*) m)[1] = m[3];      /* Bottom margin = top one. */
  746.     }
  747. #endif
  748.  
  749.     break;
  750.  
  751.     /*NOTREACHED*/
  752.  
  753.     /*
  754.      * The margins must be set so that the resulting page length will be 
  755.      * expressed exactly as a multiple of tenthes of inches.
  756.      *
  757.      */
  758.  
  759.     /**/ {
  760.     float _ds *bjcm = (float _ds*) m;
  761.  
  762.     byte pdimen = (byte)
  763.         (pdev->height / pdev->y_pixels_per_inch * 10.
  764.          - bjcm[3] * 10. - bjcm[1] * 10. + .5) + 1;
  765.     do {
  766.         --pdimen;
  767.         bjcm[1] = pdev->height / pdev->y_pixels_per_inch
  768.             - bjcm[3] - (float) pdimen / 10.;
  769.     } while (bjcm[1] < BJC_LOWER_LIMIT);
  770.     }
  771.  
  772.     break;
  773.   }
  774.   gx_device_set_margins(pdev, m, true);
  775.   return gdev_prn_open(pdev);
  776. }
  777.  
  778. /* Added parameters for DeskJet 5xxC */
  779.  
  780. /* Get parameters.  In addition to the standard and printer 
  781.  * parameters, we supply shingling and depletion parameters,
  782.  * and control over the bits-per-pixel used in output rendering */
  783. private int
  784. cdj_get_params(gx_device *pdev, gs_param_list *plist)
  785. {    int code = gdev_prn_get_params(pdev, plist);
  786.     if ( code < 0 ||
  787.         (code = param_write_int(plist, "BlackCorrect", (int *)&cdj->correction)) < 0 ||
  788.         (code = param_write_int(plist, "Shingling", &cdj->shingling)) < 0 ||
  789.         (code = param_write_int(plist, "Depletion", &cdj->depletion)) < 0
  790.        )
  791.       return code;
  792.  
  793.     return code;
  794. }
  795.  
  796. /* Put parameters. */
  797. private int
  798. cdj_put_params(gx_device *pdev, gs_param_list *plist)
  799. {    int correction = cdj->correction;
  800.     int shingling = cdj->shingling;
  801.     int depletion = cdj->depletion;
  802.     int bpp = 0;
  803.     int code = 0;
  804.  
  805.     code = cdj_put_param_int(plist, "BlackCorrect", &correction, 0, 9, code);
  806.     code = cdj_put_param_int(plist, "Shingling", &shingling, 0, 2, code);
  807.     code = cdj_put_param_int(plist, "Depletion", &depletion, 1, 3, code);
  808.     code = cdj_put_param_int(plist, "BitsPerPixel", &bpp, 1, 32, code);
  809.  
  810.     if ( code < 0 )
  811.       return code;
  812.     code = cdj_put_param_bpp(pdev, plist, bpp, bpp, 0);
  813.     if ( code < 0 )
  814.       return code;
  815.  
  816.     cdj->correction = correction;
  817.     cdj->shingling = shingling;
  818.     cdj->depletion = depletion;
  819.     return 0;
  820. }
  821.  
  822. /* Added parameters for PaintJet XL and PaintJet XL300 */
  823.  
  824. /* Get parameters.  In addition to the standard and printer
  825.  * parameters, we supply print_quality and render_type 
  826.  * parameters, together with bpp control. */
  827. private int
  828. pjxl_get_params(gx_device *pdev, gs_param_list *plist)
  829. {    int code = gdev_prn_get_params(pdev, plist);
  830.     if ( code < 0 ||
  831.         (code = param_write_int(plist, "PrintQuality", &pjxl->printqual)) < 0 ||
  832.         (code = param_write_int(plist, "RenderType", &pjxl->rendertype)) < 0
  833.        )
  834.       return code;
  835.  
  836.     return code;
  837. }
  838.  
  839. /* Put parameters. */
  840. private int
  841. pjxl_put_params(gx_device *pdev, gs_param_list *plist)
  842. {    int printqual = pjxl->printqual;
  843.     int rendertype = pjxl->rendertype;
  844.     int bpp = 0, real_bpp = 0;
  845.     int code = 0;
  846.  
  847.     code = cdj_put_param_int(plist, "PrintQuality", &printqual, -1, 1, code);
  848.     code = cdj_put_param_int(plist, "RenderType", &rendertype, 0, 10, code);
  849.     code = cdj_put_param_int(plist, "BitsPerPixel", &bpp, 1, 32, code);
  850.  
  851.     if ( code < 0 )
  852.       return code;
  853.     real_bpp = bpp;
  854.     if ( rendertype > 0 )
  855.       {    /* If printer is doing the dithering, we must have a
  856.          * true-colour mode, ie. 16 or 24 bits per pixel */
  857.         if ( bpp > 0 && bpp < 16 )
  858.           real_bpp = 24;
  859.       }
  860.     code = cdj_put_param_bpp(pdev, plist, bpp, real_bpp, 0);
  861.     if ( code < 0 )
  862.       return code;
  863.  
  864.     pjxl->printqual = printqual;
  865.     pjxl->rendertype = rendertype;
  866.     return 0;
  867. }
  868.  
  869. /* Added parameters for PaintJet */
  870.  
  871. /* Put parameters.  In addition to the standard and printer */
  872. /* parameters, we allow control of the bits-per-pixel */
  873. private int
  874. pj_put_params(gx_device *pdev, gs_param_list *plist)
  875. {    int bpp = 0;
  876.     int code = cdj_put_param_int(plist, "BitsPerPixel", &bpp, 1, 32, 0);
  877.  
  878.     if ( code < 0 )
  879.       return code;
  880.     return cdj_put_param_bpp(pdev, plist, bpp, bpp, 0);
  881. }
  882.  
  883. private stringParamDescription bjc_processColorsStrings[] = {
  884.     { "DeviceGray",        1 },
  885.     { "DeviceRGB",        3 },
  886.     { "DeviceCMYK",        4 },
  887.     { 0 }
  888. };
  889.  
  890. private stringParamDescription bjc_mediaTypeStrings[] = {
  891.     { "PlainPaper",        BJC_MEDIA_PLAINPAPER },
  892.     { "CoatedPaper",        BJC_MEDIA_COATEDPAPER },
  893.     { "TransparencyFilm",    BJC_MEDIA_TRANSPARENCYFILM },
  894.     { "Envelope",        BJC_MEDIA_ENVELOPE },
  895.     { "Card",            BJC_MEDIA_CARD},
  896.     { "Other",                BJC_MEDIA_OTHER },
  897.     { 0 }
  898. };
  899.  
  900. private stringParamDescription bjc600_printQualityStrings[] = {
  901.     { "Normal",    0 },
  902.     { "High",    1 },
  903.     { "Draft",  2 },
  904.     { 0 }
  905. };
  906.  
  907. private stringParamDescription bjc800_printQualityStrings[] = {
  908.     { "Normal",        0 },
  909.     { "High",        1 },
  910.     { "Low",        3 },
  911.     { "Draft",        4 },
  912.     { 0 },
  913. };
  914.  
  915. private stringParamDescription bjc_ditheringTypeStrings[] = {
  916.     { "None",            BJC_DITHER_NONE },
  917.     { "Floyd-Steinberg",    BJC_DITHER_FS },
  918.     { 0 }
  919. };
  920.  
  921. private int
  922. bjc_get_params(gx_device *pdev, gs_param_list *plist)
  923. {
  924.     int code = gdev_prn_get_params(pdev, plist);
  925.     int ncode;
  926.  
  927.     gs_param_string pmedia;
  928.     gs_param_string pquality;
  929.     gs_param_string dithering;
  930.  
  931.     if (code < 0) return_error(code);
  932.  
  933.     if ((ncode = param_write_bool(plist, BJC_OPTION_MANUALFEED,
  934.     &bjcparams.manualFeed)) < 0) {
  935.     code = ncode;
  936.     }
  937.  
  938.     code = get_param_string(plist, (unsigned char *)BJC_OPTION_MEDIATYPE, &pmedia,
  939.         bjc_mediaTypeStrings, bjcparams.mediaType, true, code);
  940.  
  941.     code = get_param_string(plist, (unsigned char *)BJC_OPTION_PRINTQUALITY, &pquality,
  942.         (bjc->ptype == BJC800 ? bjc800_printQualityStrings :
  943.     bjc600_printQualityStrings), bjcparams.printQuality,
  944.     true, code);
  945.  
  946.     code = get_param_string(plist, (unsigned char *)BJC_OPTION_DITHERINGTYPE, &dithering,
  947.         bjc_ditheringTypeStrings, bjcparams.ditheringType, true, code);
  948.  
  949.     if ((ncode = param_write_int(plist, BJC_OPTION_PRINTCOLORS,
  950.     &bjcparams.printColors)) < 0) {
  951.     code = ncode;
  952.     }
  953.  
  954.     if ((ncode = (bjcparams.mediaWeight_isSet ?
  955.     param_write_int(plist, BJC_OPTION_MEDIAWEIGHT,
  956.         &bjcparams.mediaWeight) :
  957.     param_write_null(plist, BJC_OPTION_MEDIAWEIGHT))) < 0) {
  958.     code = ncode;
  959.     }
  960.  
  961.     if (bjc->ptype != BJC800) {
  962.     if ((ncode = param_write_bool(plist, BJC_OPTION_MONOCHROMEPRINT,
  963.         &bjc600params.monochromePrint)) < 0) {
  964.         code = ncode;
  965.     }
  966.     }
  967.  
  968.     /**/ {
  969.     float version;
  970.     gs_param_string versionString;
  971.  
  972.     bool bTrue = true;
  973.  
  974.     version = bjcversion(pdev);
  975.     versionString.data = (byte *)bjcversionstring(pdev);
  976.             
  977.     versionString.size = strlen((char *)versionString.data);
  978.     versionString.persistent = true;
  979.  
  980.     if ((ncode = param_write_float(plist, BJC_DEVINFO_VERSION,
  981.         &version)) < 0) {
  982.         code = ncode;
  983.     }
  984.     if ((ncode = param_write_string(plist, BJC_DEVINFO_VERSIONSTRING,
  985.         &versionString)) < 0) {
  986.         code = ncode;
  987.     }
  988.  
  989.     if ((ncode = param_write_bool(plist, BJC_DEVINFO_OUTPUTFACEUP,
  990.         &bTrue)) < 0) {
  991.         code = ncode;
  992.      }
  993.     }
  994.     
  995.     return code;
  996. }
  997.  
  998. /* Put properties for the bjc drivers. */
  999.  
  1000. private int
  1001. bjc_put_params(gx_device *pdev, gs_param_list *plist)
  1002. {
  1003.     int bpp = 0, ccomps = 0;
  1004.  
  1005.     int code = 0;
  1006.     int ncode;
  1007.  
  1008.     bool aBool = true;
  1009.  
  1010.     const char* oname = (const char*) 0;
  1011.  
  1012.     bjc600_params new600Params;
  1013.     bjc800_params new800Params;
  1014.  
  1015.     bjc_params* params;
  1016.  
  1017.     gs_param_string pprocesscolors;
  1018.     gs_param_string pmedia;
  1019.     gs_param_string pquality;
  1020.  
  1021.     gs_param_float_array hwra;
  1022.  
  1023.     if (bjc->ptype != BJC800) {
  1024.     new600Params = bjc600params;
  1025.     params = (bjc_params*) &new600Params;
  1026.     } else {
  1027.     new800Params = bjc800params;
  1028.     params = (bjc_params*) &new800Params;
  1029.     }
  1030.  
  1031.     if ((code = cdj_put_param_int(plist, "BitsPerPixel",
  1032.     &bpp, 1, 32, code)) != 1) {
  1033.         bpp = pdev->color_info.depth;
  1034.     }
  1035.  
  1036.     if ((code = put_param_string(plist, (unsigned char *)"ProcessColorModel",
  1037.     &pprocesscolors, bjc_processColorsStrings, &ccomps, code)) != 1) {
  1038.         ccomps = pdev->color_info.num_components;
  1039.     }
  1040.  
  1041.     if ((ncode = param_read_bool(plist, oname = BJC_OPTION_MANUALFEED,
  1042.     ¶ms->manualFeed)) < 0) {
  1043.     param_signal_error(plist, oname, code = ncode);
  1044.     }
  1045.  
  1046.     code = put_param_string(plist, (unsigned char *)BJC_OPTION_MEDIATYPE, &pmedia,
  1047.         bjc_mediaTypeStrings, ¶ms->mediaType, code);
  1048.  
  1049.     code = cdj_put_param_int(plist, BJC_OPTION_PRINTCOLORS,
  1050.     ¶ms->printColors, 0, 15, code);
  1051.  
  1052.     code = put_param_string(plist, (unsigned char *)BJC_OPTION_PRINTQUALITY, &pquality,
  1053.     (bjc->ptype == BJC800 ? bjc800_printQualityStrings :
  1054.     bjc600_printQualityStrings), ¶ms->printQuality, code);
  1055.  
  1056.     switch (ncode = param_read_int(plist,
  1057.     oname = BJC_OPTION_MEDIAWEIGHT, ¶ms->mediaWeight)) {
  1058.     case 0:
  1059.         if (params->mediaWeight <= 0) {
  1060.         ncode = gs_error_rangecheck;
  1061.         } else {
  1062.         params->mediaWeight_isSet = 1;
  1063.         break;
  1064.         }
  1065.         goto mwe;
  1066.  
  1067.     default:
  1068.         if ((ncode = param_read_null(plist, oname)) == 0) {
  1069.         params->mediaWeight_isSet = 0;
  1070.         break;
  1071.         }
  1072. mwe:           param_signal_error(plist, oname, code = ncode);
  1073.  
  1074.     case 1:
  1075.         break;
  1076.     }
  1077.  
  1078.     if (bjc->ptype != BJC800) {
  1079.     bjc600_params* params600 = (bjc600_params*) params;
  1080.     if ((ncode = param_read_bool(plist,
  1081.         oname = BJC_OPTION_MONOCHROMEPRINT,
  1082.         ¶ms600->monochromePrint)) < 0) {
  1083.         param_signal_error(plist, oname, code = ncode);
  1084.     }
  1085.     }
  1086.  
  1087.     if ((ncode = cdj_param_check_float(plist, BJC_DEVINFO_VERSION,
  1088.     bjcversion(pdev), true)) < 0) {
  1089.     code = ncode;
  1090.     }
  1091.     if ((ncode = cdj_param_check_string(plist, BJC_DEVINFO_VERSIONSTRING,
  1092.     bjcversionstring(pdev), true)) < 0) {
  1093.     code = ncode;
  1094.     }
  1095.  
  1096.     if ((ncode = param_read_bool(plist, oname = BJC_DEVINFO_OUTPUTFACEUP,
  1097.     &aBool)) < 0) {
  1098.     param_signal_error(plist, oname, code = ncode);
  1099.     } else if (aBool != true) {
  1100.     param_signal_error(plist, oname, code = ncode = gs_error_rangecheck);
  1101.     }
  1102.  
  1103.     /* Check for invalid resolution. The array macros are taken from
  1104.        gsdparam.c and modified to use oname, ncode and code instead
  1105.        of param_name, code and ecode respectively. */
  1106.  
  1107.     BEGIN_ARRAY_PARAM(param_read_float_array, "HWResolution", hwra, 2, hwre)
  1108.     if ( hwra.data[0] <= 0 || hwra.data[1] <= 0 ||
  1109.          hwra.data[0] != hwra.data[1] )
  1110.         ncode = gs_error_rangecheck;
  1111.     else {
  1112. #ifdef BJC_STRICT
  1113.         if (hwra.data[0] != BJC_RESOLUTION_LOW &&
  1114.         hwra.data[0] != BJC_RESOLUTION_NORMAL &&
  1115.         hwra.data[0] != BJC_RESOLUTION_HIGH) {
  1116.         ncode = gs_error_rangecheck; 
  1117.         }
  1118. #else
  1119.         /* A small hack for checking resolution without logarithms. */
  1120.  
  1121.         /**/ {
  1122.         int n;
  1123.  
  1124.         for (n = 0; n < 8 * sizeof(n) / BJC_RESOLUTION_BASE; ++n) {
  1125.             float res = BJC_RESOLUTION_BASE * (1 << n);
  1126.         
  1127.             if (res == hwra.data[0]) break;
  1128.         
  1129.             if (res > hwra.data[0]) {
  1130.             ncode = gs_error_rangecheck;
  1131.             }
  1132.         }
  1133.  
  1134.         if (n == 8 * sizeof(n)) {
  1135.             ncode = gs_error_rangecheck;
  1136.         }
  1137.         }
  1138. #endif
  1139.         if (ncode < 0) {
  1140.         code = ncode;
  1141.         } else {
  1142.         break;
  1143.         }
  1144.     }
  1145.     END_PARAM(hwra, hwre)
  1146.  
  1147.     if ((ncode = cdj_put_param_bpp(pdev, plist, bpp, bpp, ccomps)) < 0) {
  1148.     code = ncode;
  1149.     }
  1150.  
  1151.     if (code < 0)
  1152.     return code;
  1153.  
  1154.     if (bpp == 1) {
  1155.     params->ditheringType = BJC_DITHER_NONE;
  1156.     }
  1157.  
  1158.     /* Write values that did change */
  1159.  
  1160.     if (bjc->ptype != BJC800) {
  1161.     bjc600params = new600Params;
  1162.     } else {
  1163.     bjc800params = new800Params;
  1164.     }
  1165.  
  1166.     return code;
  1167. }
  1168.  
  1169. /* ------ Internal routines ------ */
  1170.  
  1171. /* The DeskJet500C can compress (mode 9) */
  1172. private int
  1173. dj500c_print_page(gx_device_printer * pdev, FILE * prn_stream)
  1174. {
  1175.   return hp_colour_print_page(pdev, prn_stream, DJ500C);
  1176. }
  1177.  
  1178. /* The DeskJet550C can compress (mode 9) */
  1179. private int
  1180. dj550c_print_page(gx_device_printer * pdev, FILE * prn_stream)
  1181. {
  1182.   return hp_colour_print_page(pdev, prn_stream, DJ550C);
  1183. }
  1184.  
  1185. /* The DesignJet650C can compress (mode 1) */
  1186. private int
  1187. dnj650c_print_page(gx_device_printer * pdev, FILE * prn_stream)
  1188. {
  1189.   return hp_colour_print_page(pdev, prn_stream, DNJ650C);
  1190. }
  1191.  
  1192. private int
  1193. lj4dith_print_page(gx_device_printer * pdev, FILE * prn_stream)
  1194. {
  1195.   return hp_colour_print_page(pdev, prn_stream, LJ4DITH);
  1196. }
  1197.  
  1198. /* The PJXL300 can compress (modes 2 & 3) */
  1199. private int
  1200. pjxl300_print_page(gx_device_printer * pdev, FILE * prn_stream)
  1201. { int ret_code;
  1202.   /* Ensure we're operating in PCL mode */
  1203.   fputs("\033%-12345X@PJL enter language = PCL\n", prn_stream);
  1204.   ret_code = hp_colour_print_page(pdev, prn_stream, PJXL300);
  1205.   /* Reenter switch-configured language */
  1206.   fputs("\033%-12345X", prn_stream);
  1207.   return ret_code;
  1208. }
  1209.  
  1210. /* The PaintJet XL can compress (modes 2 & 3) */
  1211. private int
  1212. pjxl_print_page(gx_device_printer * pdev, FILE * prn_stream)
  1213. {
  1214.   return hp_colour_print_page(pdev, prn_stream, PJXL180);
  1215. }
  1216.  
  1217. /* The PaintJet can compress (mode 1) */
  1218. private int
  1219. pj_print_page(gx_device_printer * pdev, FILE * prn_stream)
  1220. {
  1221.   return hp_colour_print_page(pdev, prn_stream, PJ180);
  1222. }
  1223.  
  1224. /* The LJ250 can compress (mode 1) */
  1225. private int
  1226. declj250_print_page(gx_device_printer * pdev, FILE * prn_stream)
  1227. { int ret_code;
  1228.   fputs("\033%8", prn_stream);    /* Enter PCL emulation mode */
  1229.   ret_code = hp_colour_print_page(pdev, prn_stream, DECLJ250);
  1230.   fputs("\033%@", prn_stream);    /* Exit PCL emulation mode */
  1231.   return ret_code;
  1232. }
  1233.  
  1234. /* The BJC-600 cannot compress w/o raster image commands. */
  1235. private int
  1236. escp_print_page(gx_device_printer * pdev, FILE * prn_stream)
  1237. {
  1238.   return hp_colour_print_page(pdev, prn_stream, ESC_P);
  1239. }
  1240.  
  1241. /* The BJC-600 can compress w/ raster image commands. */
  1242. private int
  1243. bjc_print_page(gx_device_printer * pdev, FILE * prn_stream)
  1244. {
  1245.   return hp_colour_print_page(pdev, prn_stream, bjc->ptype);
  1246. }
  1247.  
  1248. /* MACROS FOR DITHERING (we use macros for compact source and faster code) */
  1249. /* Floyd-Steinberg dithering. Often results in a dramatic improvement in
  1250.  * subjective image quality, but can also produce dramatic increases in
  1251.  * amount of printer data generated and actual printing time!! Mode 9 2D
  1252.  * compression is still useful for fairly flat colour or blank areas but its
  1253.  * compression is much less effective in areas where the dithering has
  1254.  * effectively randomised the dot distribution. */
  1255.  
  1256. #define SHIFT ((I * 8) - 13)
  1257. #define RSHIFT ((I * 8) - 16)
  1258. #define RANDOM (((rand() << RSHIFT) % (MAXVALUE / 2))  - MAXVALUE / 4);
  1259. #define MINVALUE  0
  1260. #define MAXVALUE  (255 << SHIFT)
  1261. #define THRESHOLD (128 << SHIFT)
  1262. #define C 8
  1263.  
  1264. #define FSdither(inP, out, errP, Err, Bit, Offset, Element)\
  1265.     oldErr = Err;\
  1266.     Err = (errP[Element] + ((Err * 7 + C) >> 4) + ((int)inP[Element] << SHIFT));\
  1267.     if (Err > THRESHOLD) {\
  1268.       out |= Bit;\
  1269.       Err -= MAXVALUE;\
  1270.     }\
  1271.     errP[Element + Offset] += ((Err * 3 + C) >> 4);\
  1272.     errP[Element] = ((Err * 5 + oldErr + C) >> 4);
  1273.  
  1274. /* Here we rely on compiler optimisation to remove lines of the form
  1275.  * (if (1 >= 4) {...}, ie. the constant boolean expressions */
  1276.  
  1277. /* The original code is in the #else part. Since by default NEW_DITHER
  1278.    is not defined, the old code is used. No enhancement is visible for the
  1279.    bjc600 drivers with the new code, anyway :-( */
  1280.  
  1281. #ifdef NEW_DITHER
  1282.  
  1283. #define FSDline(scan, i, j, plane_size, cErr, mErr, yErr, kErr, cP, mP, yP, kP, n)\
  1284. {\
  1285.     if (scan == 0) {       /* going_up */\
  1286.       for (i = 0; i < plane_size; i++) {\
  1287.     byte c, y, m, k, bitmask;\
  1288.     int oldErr;\
  1289.     bitmask = 0x80;\
  1290.     for (c = m = y = k = 0; bitmask != 0; bitmask >>= 1) {\
  1291.       if (n >= 4) {\
  1292.           FSdither(dp, k, ep, kErr, bitmask, -n, 0);\
  1293.       }\
  1294.       if (n >= 3) {\
  1295.           FSdither(dp, c, ep, cErr, bitmask, -n, n - 3);\
  1296.           FSdither(dp, m, ep, mErr, bitmask, -n, n - 2);\
  1297.       }\
  1298.       FSdither(dp, y, ep, yErr, bitmask, -n, n - 1);\
  1299.       dp += n, ep += n;\
  1300.     }\
  1301.     if (n >= 4)\
  1302.       *kP++ = k;\
  1303.     if (n >= 3) {\
  1304.       *cP++ = c;\
  1305.           *mP++ = m;\
  1306.     }\
  1307.     *yP++ = y;\
  1308.       }\
  1309.     } else {        /* going_down */\
  1310.       for (i = 0; i < plane_size; i++) {\
  1311.     byte c, y, m, k, bitmask;\
  1312.     int oldErr;\
  1313.     bitmask = 0x01;\
  1314.     for (c = m = y = k = 0; bitmask != 0; bitmask <<= 1) {\
  1315.           dp -= n, ep -= n;\
  1316.       FSdither(dp, y, ep, yErr, bitmask, n, n - 1);\
  1317.       if (n >= 3) {\
  1318.         FSdither(dp, m, ep, mErr, bitmask, n, n - 2);\
  1319.         FSdither(dp, c, ep, cErr, bitmask, n, n - 3);\
  1320.       }\
  1321.       if (n >= 4) {\
  1322.         FSdither(dp, k, ep, kErr, bitmask, n, 0);\
  1323.       }\
  1324.     }\
  1325.     *--yP = y;\
  1326.     if (n >= 3)\
  1327.       { *--mP = m;\
  1328.         *--cP = c;\
  1329.       }\
  1330.     if (n >= 4)\
  1331.       *--kP = k;\
  1332.       }\
  1333.     }\
  1334. }
  1335.  
  1336. #else
  1337.  
  1338. #define FSDline(scan, i, j, plane_size, cErr, mErr, yErr, kErr, cP, mP, yP, kP, n)\
  1339. {\
  1340.     if (scan == 0) {       /* going_up */\
  1341.       for (i = 0; i < plane_size; i++) {\
  1342.     byte c, y, m, k, bitmask;\
  1343.     int oldErr;\
  1344.     bitmask = 0x80;\
  1345.     for (c = m = y = k = 0; bitmask != 0; bitmask >>= 1) {\
  1346.       if (n >= 4) {\
  1347.         if (*dp) {\
  1348.           FSdither(dp, k, ep, kErr, bitmask, -n, 0);\
  1349.           cErr = mErr = yErr = 0;\
  1350.         } else {\
  1351.           FSdither(dp, c, ep, cErr, bitmask, -n, n - 3);\
  1352.           FSdither(dp, m, ep, mErr, bitmask, -n, n - 2);\
  1353.           FSdither(dp, y, ep, yErr, bitmask, -n, n - 1);\
  1354.         }\
  1355.       } else {\
  1356.         if (n >= 3) {\
  1357.           FSdither(dp, c, ep, cErr, bitmask, -n, n - 3);\
  1358.           FSdither(dp, m, ep, mErr, bitmask, -n, n - 2);\
  1359.         }\
  1360.         FSdither(dp, y, ep, yErr, bitmask, -n, n - 1);\
  1361.       }\
  1362.       dp += n, ep += n;\
  1363.     }\
  1364.     if (n >= 4)\
  1365.       *kP++ = k;\
  1366.     if (n >= 3) {\
  1367.       *cP++ = c;\
  1368.           *mP++ = m;\
  1369.     }\
  1370.     *yP++ = y;\
  1371.       }\
  1372.     } else {        /* going_down */\
  1373.       for (i = 0; i < plane_size; i++) {\
  1374.     byte c, y, m, k, bitmask;\
  1375.     int oldErr;\
  1376.     bitmask = 0x01;\
  1377.     for (c = m = y = k = 0; bitmask != 0; bitmask <<= 1) {\
  1378.           dp -= n, ep -= n;\
  1379.       if (n >= 4) {\
  1380.             if (*dp) {\
  1381.           FSdither(dp, k, ep, kErr, bitmask, n, 0);\
  1382.           cErr = mErr = yErr = 0;\
  1383.         } else {\
  1384.           FSdither(dp, y, ep, yErr, bitmask, n, n - 1);\
  1385.           FSdither(dp, m, ep, mErr, bitmask, n, n - 2);\
  1386.           FSdither(dp, c, ep, cErr, bitmask, n, n - 3);\
  1387.         }\
  1388.       } else {\
  1389.         FSdither(dp, y, ep, yErr, bitmask, n, n - 1);\
  1390.         if (n >= 3) {\
  1391.           FSdither(dp, m, ep, mErr, bitmask, n, n - 2);\
  1392.           FSdither(dp, c, ep, cErr, bitmask, n, n - 3);\
  1393.         }\
  1394.       }\
  1395.     }\
  1396.     *--yP = y;\
  1397.     if (n >= 3)\
  1398.       { *--mP = m;\
  1399.         *--cP = c;\
  1400.       }\
  1401.     if (n >= 4)\
  1402.       *--kP = k;\
  1403.       }\
  1404.     }\
  1405. }
  1406.  
  1407. #endif
  1408.  
  1409. /* END MACROS FOR DITHERING */
  1410.  
  1411. #define CPbit(inP, out, Bit, Element)\
  1412.   if (inP[Element]) {\
  1413.     out |= Bit;\
  1414.   }
  1415.  
  1416. #define COPYline(scan, i, j, plane_size, cP, mP, yP, kP, n)\
  1417. {\
  1418.     if (scan == 0) {       /* going_up */\
  1419.       for (i = 0; i < plane_size; i++) {\
  1420.     byte c, y, m, k, bitmask;\
  1421.     bitmask = 0x80;\
  1422.     for (c = m = y = k = 0; bitmask != 0; bitmask >>= 1) {\
  1423.       if (n >= 4) {\
  1424.         CPbit(dp, k, bitmask, 0);\
  1425.       } \
  1426.           if (n >= 3) {\
  1427.         CPbit(dp, c, bitmask, n - 3);\
  1428.         CPbit(dp, m, bitmask, n - 2);\
  1429.       }\
  1430.       CPbit(dp, y, bitmask, n - 1);\
  1431.       dp += n, ep += n;\
  1432.     }\
  1433.     if (n >= 4)\
  1434.       *kP++ = k;\
  1435.     if (n >= 3) {\
  1436.       *cP++ = c;\
  1437.           *mP++ = m;\
  1438.     }\
  1439.     *yP++ = y;\
  1440.       }\
  1441.     } else {        /* going_down */\
  1442.       for (i = 0; i < plane_size; i++) {\
  1443.     byte c, y, m, k, bitmask;\
  1444.     bitmask = 0x01;\
  1445.     for (c = m = y = k = 0; bitmask != 0; bitmask <<= 1) {\
  1446.           dp -= n, ep -= n;\
  1447.       if (n >= 4) {\
  1448.             CPbit(dp, k, bitmask, 0);\
  1449.       }\
  1450.       if (n >= 3) {\
  1451.         CPbit(dp, m, bitmask, n - 2);\
  1452.         CPbit(dp, c, bitmask, n - 3);\
  1453.       }\
  1454.       CPbit(dp, y, bitmask, n - 1);\
  1455.     }\
  1456.     *--yP = y;\
  1457.     if (n >= 3)\
  1458.       { *--mP = m;\
  1459.         *--cP = c;\
  1460.       }\
  1461.     if (n >= 4)\
  1462.       *--kP = k;\
  1463.       }\
  1464.     }\
  1465. }
  1466.  
  1467. /* Some convenient shorthand .. */
  1468. #define x_dpi        (pdev->x_pixels_per_inch)
  1469. #define y_dpi        (pdev->y_pixels_per_inch)
  1470. #define CONFIG_16BIT "\033*v6W\000\003\000\005\006\005"
  1471. #define CONFIG_24BIT "\033*v6W\000\003\000\010\010\010"
  1472.  
  1473. /* To calculate buffer size as next greater multiple of both parameter and W */
  1474. #define calc_buffsize(a, b) (((((a) + ((b) * W) - 1) / ((b) * W))) * W)
  1475.  
  1476.  
  1477. /*
  1478.  * Miscellaneous functions for Canon BJC-600 printers in raster command mode.
  1479.  */
  1480. #define fputshort(n, f) fputc((n)%256,f);fputc((n)/256,f)
  1481.  
  1482. private int
  1483. bjc_cmd(byte cmd, int argsize, byte* arg, gx_device_printer* pdev,
  1484.     FILE* stream)
  1485. {
  1486.   fputs("\033(", stream);
  1487.   putc(cmd, stream);
  1488.   fputshort(argsize, stream);
  1489.   fwrite(arg, sizeof(byte), argsize, stream);
  1490.  
  1491.   return 0;
  1492. }
  1493.  
  1494.  
  1495. private int
  1496. bjc_raster_cmd_sub(char c, int rastsize, byte* data, FILE* stream)
  1497. {
  1498.   fputs("\033(A", stream);
  1499.   fputshort(rastsize + 1, stream);
  1500.   putc(c, stream);
  1501.   fwrite(data, sizeof(byte), rastsize, stream);
  1502.   putc('\015', stream);
  1503.  
  1504.   return 0;
  1505. }
  1506.  
  1507. private int
  1508. bjc_raster_cmd(int c_id, int rastsize, byte* data, gx_device_printer* pdev,
  1509.     FILE* stream)
  1510. {
  1511.     if (bjcparams.printColors == BJC_COLOR_ALLBLACK) {
  1512.     bjc_raster_cmd_sub('K', rastsize, data, stream);
  1513.     } else if (pdev->color_info.num_components == 1) {
  1514.     if (bjcparams.printColors & BJC_COLOR_BLACK) {
  1515.         bjc_raster_cmd_sub('K', rastsize, data, stream);
  1516.     } else {
  1517.         if (bjcparams.printColors & BJC_COLOR_YELLOW)
  1518.         bjc_raster_cmd_sub('Y', rastsize, data, stream);
  1519.         if (bjcparams.printColors & BJC_COLOR_MAGENTA)
  1520.         bjc_raster_cmd_sub('M', rastsize, data, stream);
  1521.         if (bjcparams.printColors & BJC_COLOR_CYAN)
  1522.         bjc_raster_cmd_sub('C', rastsize, data, stream);
  1523.     }
  1524.     }else {            /* Color decomposition */
  1525.     private byte ymckCodes[] = {
  1526.         BJC_COLOR_YELLOW,
  1527.         BJC_COLOR_MAGENTA,
  1528.         BJC_COLOR_CYAN,
  1529.         BJC_COLOR_BLACK,
  1530.     };
  1531.  
  1532.     if (bjcparams.printColors & (int) ymckCodes[c_id]) {
  1533.         bjc_raster_cmd_sub("YMCK"[c_id], rastsize, data, stream);
  1534.     }
  1535.     }
  1536.  
  1537.     return 0;
  1538. }
  1539.  
  1540. private int
  1541. bjc_init_page(gx_device_printer* pdev, FILE* stream)
  1542. {
  1543.     byte pagemargins[3], resolution[2], paperloading[2];
  1544.  
  1545.     /* Compute page margins. */
  1546.  
  1547.     pagemargins[0] = (byte) ((float) pdev->height / pdev->y_pixels_per_inch
  1548.         * 10 + .5);
  1549.     pagemargins[1] = (byte) 1;
  1550.     pagemargins[2] = (byte) ((pdev->width / pdev->x_pixels_per_inch * 10) -
  1551.         pdev->HWMargins[0] / 7.2 - pdev->HWMargins[2] / 7.2 + .5);
  1552.  
  1553.     /* Cheat to keep margins into bounds (while waiting to have the right
  1554.        margins for big papers. */
  1555.  
  1556.     switch (bjc->ptype) {
  1557.     case BJC800:
  1558.         if (pagemargins[2] > 114) pagemargins[2] = 114;
  1559.         break;
  1560.  
  1561.     default:
  1562.         if (pagemargins[2] > 80) pagemargins[2] = 80;
  1563.         break;
  1564.     }
  1565.  
  1566.     /* Initialize resolution argument. */
  1567.  
  1568.     resolution[0] = (byte) ((int)pdev->x_pixels_per_inch / 256);
  1569.     resolution[1] = (byte) ((int)pdev->x_pixels_per_inch % 256);
  1570.  
  1571.     /* Initialize paper loading argument. */
  1572.  
  1573.     paperloading[0] = 0x10 + ((1 - bjcparams.manualFeed) << 2);
  1574.     paperloading[1] = bjcparams.mediaType << 4;
  1575.  
  1576.     /* Reinitialize printer in raster mode. */
  1577.  
  1578.     fputs("\033[K", stream);
  1579.     fputshort(2, stream);
  1580.     fputc(0x00, stream);
  1581.     fputc(0x0f, stream);
  1582.  
  1583.     /* Set page mode on (ignore data at end of page) */
  1584.  
  1585.     bjc_cmd('a', 1, (byte*) "\001", pdev, stream);
  1586.  
  1587.     /* Set page margins */
  1588.  
  1589.     bjc_cmd('g', 3, pagemargins, pdev, stream);
  1590.  
  1591.     /* Set compression on (this is PackBits compression a la TIFF/Mac) */
  1592.  
  1593.     bjc_cmd('b', 1, (byte*) "\001", pdev, stream);
  1594.  
  1595.     /* Set paper loading. */
  1596.  
  1597.     bjc_cmd('l', 2, paperloading, pdev, stream);
  1598.  
  1599.     /* Set printing method. */
  1600.  
  1601. #ifndef BJC_INIT_800_AS_600
  1602.     if (bjc->ptype == BJC800) {
  1603. #else
  1604.     if (0) {
  1605. #endif
  1606.     byte printmode[2];
  1607.  
  1608.     printmode[0] = bjcparams.printQuality;
  1609.  
  1610.     /* Modes not used are 3 (CN, Color Normal) and 2 (TP+ (?)) */
  1611.  
  1612.     switch (bjcparams.printQuality) {
  1613.         case BJC_QUALITY_DRAFT:
  1614.             printmode[0] = 4;            /* Draft */
  1615.         break;
  1616.     }
  1617.  
  1618.     printmode[1] = (bjcparams.mediaType >= BJC_MEDIA_ENVELOPE ? 1 :
  1619.         bjc800thickpaper());
  1620.  
  1621.     bjc_cmd('c', 2, printmode, pdev, stream);
  1622.     } else /* BJC600 */ {
  1623.     byte printmeth[3];
  1624.  
  1625.     printmeth[0] = 0x10 + ((1 - bjcparams.manualFeed) << 2);
  1626.     printmeth[1] = (bjcparams.mediaType << 4) + bjcparams.printQuality;
  1627.     printmeth[2] = (bjcparams.printQuality == BJC_QUALITY_HIGH ?
  1628.         0x10 : 0) + (bjcparams.mediaType >= BJC_MEDIA_ENVELOPE ? 1 :
  1629.              bjc600thickpaper());
  1630.  
  1631.         bjc_cmd('c', 3, printmeth, pdev, stream);
  1632.     }
  1633.  
  1634.     /* Set raster resolution */
  1635.  
  1636.     bjc_cmd('d', 2, resolution, pdev, stream);
  1637.  
  1638.     return 0;
  1639. }
  1640.  
  1641. private int
  1642. bjc_v_skip(int n, gx_device_printer* pdev, FILE* stream)
  1643. {
  1644.     if (n) {
  1645.     fputs("\033(e", stream);
  1646.     putc(2, stream);
  1647.     putc(0, stream);
  1648.     putc(n / 256, stream);
  1649.     putc(n % 256, stream);
  1650.     }
  1651.  
  1652.     return 0;
  1653. }
  1654.  
  1655. private int
  1656. bjc_finish_page(gx_device_printer* pdev, FILE* stream)
  1657. {
  1658.     bjc_cmd('a', 1, (byte*) "\000", pdev, stream);
  1659.     bjc_cmd('b', 1, (byte*) "\000", pdev, stream);
  1660.     fputc('\014', stream);
  1661.     fputs("\033@", stream);
  1662.  
  1663.     return 0;
  1664. }
  1665.  
  1666. /* 1D runlength compression for BJC-600
  1667.  * this code is borrowed from gdevpcl.c:gdev_pcl_mode2compress.
  1668.  */
  1669. private int
  1670. bjc_compress(const byte *row, const byte *end_row, byte *compressed)
  1671. {
  1672.   register const byte *exam = row;
  1673.   register byte *cptr = compressed; /* output pointer into compressed bytes */
  1674.   
  1675.  
  1676.   while ( exam < end_row ) {
  1677.     /* Search ahead in the input looking for a run */
  1678.     /* of at least 4 identical bytes. */
  1679.     const byte *compr = exam;
  1680.     const byte *end_dis;
  1681.     const byte *next;
  1682.     register byte test, test2;
  1683.  
  1684.     test = *exam;
  1685.     while ( exam < end_row ) {
  1686.       test2 = *++exam;
  1687.       if ( test == test2 )
  1688.       break;
  1689.       test = test2;
  1690.     }
  1691.     
  1692.  
  1693.     /* Find out how long the run is */
  1694.     end_dis = exam - 1;
  1695.     if ( exam == end_row ) { /* no run */
  1696.       next = --end_row;
  1697.     } else { 
  1698.  
  1699.       next = exam + 1;
  1700.       while ( next < end_row && *next == test ) next++;
  1701.     }
  1702.     
  1703.  
  1704.     /* Now [compr..end_dis) should be encoded as dissimilar, */
  1705.     /* and [end_dis..next) should be encoded as similar. */
  1706.     /* Note that either of these ranges may be empty. */
  1707.     
  1708.  
  1709.     for ( ; ; ) {    /* Encode up to 128 dissimilar bytes */
  1710.       uint count = end_dis - compr; /* uint for faster switch */
  1711.       switch ( count ) { /* Use memcpy only if it's worthwhile. */
  1712.       case 6: cptr[6] = compr[5];
  1713.       case 5: cptr[5] = compr[4];
  1714.       case 4: cptr[4] = compr[3];
  1715.       case 3: cptr[3] = compr[2];
  1716.       case 2: cptr[2] = compr[1];
  1717.       case 1: cptr[1] = compr[0];
  1718.     *cptr = count - 1;
  1719.     cptr += count + 1;
  1720.       case 0: /* all done */
  1721.     break;
  1722.       default:
  1723.     if ( count > 128 ) count = 128;
  1724.     *cptr++ = count - 1;
  1725.     memcpy(cptr, compr, count);
  1726.     cptr += count, compr += count;
  1727.     continue;
  1728.       }
  1729.       break;
  1730.     }
  1731.     
  1732.  
  1733.     {    /* Encode up to 128 similar bytes. */
  1734.       /* Note that count may be <0 at end of row. */
  1735.       int count = next - end_dis;
  1736.       if (next < end_row || test != 0)
  1737.     while ( count > 0 ) { 
  1738.  
  1739.       int this = (count > 128 ? 128 : count);
  1740.       *cptr++ = 257 - this;
  1741.       *cptr++ = (byte)test;
  1742.       count -= this;
  1743.     }
  1744.       exam = next;
  1745.     }
  1746.   }
  1747.   return cptr - compressed;
  1748. }
  1749.  
  1750. /*
  1751.  * For the ESC/P mode, resolution is fixed as 360dpi and we must transform
  1752.  * image data to serialized data.
  1753.  */
  1754. private word *ep_storage;
  1755. private uint ep_storage_size_words;
  1756. private byte *ep_raster_buf[4][BJC_HEAD_ROWS], *ep_print_buf;
  1757. private int ep_num_comps, ep_plane_size, img_rows=BJC_HEAD_ROWS;
  1758.  
  1759.  
  1760. #define row_bytes (img_rows / 8)
  1761. #define row_words (row_bytes / sizeof(word))
  1762. #define min_rows (32)        /* for optimization of text image printing */
  1763.  
  1764.  
  1765. private int
  1766. ep_print_image(FILE *prn_stream, char cmd, byte *data, int size)
  1767. {
  1768.   static int ln_idx=0, vskip1=0, vskip2=0, real_rows;
  1769.   int i;
  1770.   static const char color[4] = {4,1,2,0};
  1771.  
  1772.  
  1773.   switch (cmd) {
  1774.   case 3:            /* Black */
  1775.   case 2:            /* Cyan */
  1776.   case 1:            /* Magenta */
  1777.   case 0:            /* Yellow */
  1778.     memcpy(ep_raster_buf[((int) cmd)][ln_idx+vskip2], data, size);
  1779.     return 0;
  1780.   case 'B':            /* blank line skip */
  1781.     if (!ln_idx) {
  1782.       vskip1 += size;
  1783.     } else if (size >= img_rows - (ln_idx+vskip2) || ln_idx+vskip2 >= min_rows) {
  1784.       /* The 'I' cmd must precede 'B' cmd! */
  1785.       vskip2 += size;
  1786.       ep_print_image(prn_stream, 'F', 0, 0); /* flush and reset status */
  1787.     } else {
  1788.       vskip2 += size;
  1789.     }
  1790.     return 0;
  1791.   case 'I':            /* Increment index */
  1792.     ln_idx += vskip2 + 1;
  1793.     vskip2 = 0;
  1794.     if (ln_idx < img_rows) return 0;
  1795.     /* if ep_raster_buf filled up, then fall through here and flush buffer */
  1796.   case 'F':            /* flush print buffer */
  1797.     if (!ln_idx) return 0;    /* The end of the page. */
  1798.  
  1799.  
  1800.     /* before print the image, perform vertical skip. */
  1801.     while (vskip1 >= (255*2)) {
  1802.       fputs("\033J\377", prn_stream); /* n/180in. feeding */
  1803.       vskip1 -= (255*2);
  1804.     }
  1805.     if (vskip1 > 255) {
  1806.       fputs("\033J\200", prn_stream);
  1807.       vskip1 -= 256;
  1808.     }
  1809.     if (vskip1) {
  1810.       /* n/360in. feeding */
  1811.       fputs("\033|J", prn_stream); putc(0, prn_stream); putc(vskip1, prn_stream);
  1812.     }
  1813.  
  1814.  
  1815.     /* Optimize the number of nozzles to be used. */
  1816.     if (ln_idx > 56) {        /* use 64 nozzles */
  1817.       real_rows = 64;
  1818.     } else if (ln_idx > 48) {    /* use 56 nozzles */
  1819.       real_rows = 56;
  1820.     } else if (ln_idx > 32) {    /* use 48 nozzles */
  1821.       real_rows = 48;
  1822.     } else {            /* use 32 nozzles */
  1823.       real_rows = 32;
  1824.     }
  1825.  
  1826.  
  1827.     for (i = 0; i < ep_num_comps; i++) {
  1828.       int lnum, hskip, print_size, img_rows;
  1829.       byte *p0, *p1, *p2, *p3;
  1830.       byte *inp, *inbuf, *outp, *outbuf;
  1831.  
  1832.  
  1833.       img_rows = real_rows;    /* Note that this img_rows is not the one that
  1834.                  * defined out of this function. */
  1835.       outbuf = ep_print_buf;
  1836.  
  1837.  
  1838.       /* Transpose raster image for serial printer image */
  1839.       for (lnum=0; lnum < img_rows; lnum+=8, outbuf++) {
  1840.     inbuf = inp = ep_raster_buf[i][lnum];
  1841.     for (outp = outbuf; inp < inbuf+ep_plane_size; inp++, outp += img_rows) {
  1842.       memflip8x8(inp, ep_plane_size, outp, row_bytes);
  1843.     }
  1844.       }
  1845.  
  1846.  
  1847.       /* Set color */
  1848.       if (ep_num_comps == 1) {
  1849.     /* Don't set color (to enable user setting). */
  1850.     putc('\015', prn_stream);
  1851.       } else {
  1852.     /* set color to one of CMYK. */
  1853.     fputs("\015\033r", prn_stream);
  1854.     putc(color[i], prn_stream);
  1855.       }
  1856.  
  1857.  
  1858.       *(outp = ep_print_buf + ep_plane_size * img_rows) = 1; /* sentinel */
  1859.  
  1860.  
  1861.       p0 = p3 = ep_print_buf;
  1862.  
  1863.  
  1864.       /* print image p0 to p1 and h skip p1 to p2 if p2<outp,
  1865.        * then make p0=p2 and continue */
  1866.       while (p0 < outp) {
  1867.     static const word zeros[8] = {0,0,0,0,0,0,0,0};
  1868.  
  1869.  
  1870.     if (p3 < outp) {
  1871.       /* p1 is the head of running zeros. */
  1872.       /* note that h skip unit is 1/180inch */
  1873.       for (p1 = p3; !memcmp(p3, zeros, row_bytes*2); p3 += row_bytes*2);
  1874.       /* p2 is the head of non zero image. */
  1875.       p2 = p3;
  1876.     redo:
  1877.       for (p3 += row_bytes; memcmp(p3, zeros, row_bytes); p3 += row_bytes);
  1878.       if (p3 < outp && memcmp(p3+row_bytes, zeros, row_bytes)) goto redo;
  1879.     } else p1 = p2 = outp;
  1880.  
  1881.  
  1882.     if (p0 < p1) {    /* print the image between p0 and p1 */
  1883.       print_size = ((p1 < outp) ? p1 : outp) - p0;
  1884.       fputs("\033|B", prn_stream); putc(img_rows, prn_stream);
  1885.       fputshort(print_size, prn_stream);
  1886.       fwrite(p0, sizeof(byte), print_size, prn_stream);
  1887.     }
  1888.     if (p1 < p2) {    /* skip running zeros from p1 to p2 */
  1889.       hskip = (((p2 < outp) ? p2 : outp) - p1) / row_bytes / 2;
  1890.       fputs("\033\\", prn_stream);
  1891.       fputshort(hskip, prn_stream);
  1892.     }
  1893.     p0 = p2;
  1894.       }
  1895.     }
  1896.     return ep_print_image(prn_stream, 'R', 0, vskip2 + ln_idx); 
  1897.   case 'R':            /* Reset status */
  1898.     ln_idx = 0;
  1899.     vskip1 = size;
  1900.     vskip2 = 0;
  1901.     memset(ep_storage, 0, ep_storage_size_words * W);
  1902.     return 0;
  1903.   default:            /* This should not happen */
  1904.     fprintf(stderr, "ep_print_image: illegal command character `%c'.\n", cmd);
  1905.     return 1;
  1906.   }
  1907.  
  1908.  
  1909.   /* NOT REACHED */
  1910. }
  1911.  
  1912.  
  1913. /* Send the page to the printer.  Compress each scan line. */
  1914. private int
  1915. hp_colour_print_page(gx_device_printer * pdev, FILE * prn_stream, int ptype)
  1916. {
  1917.   uint raster_width = gdev_prn_rasterwidth(pdev, 1);
  1918. /*  int line_size = gdev_prn_rasterwidth(pdev, 0); */
  1919.   int line_size = gdev_prn_raster(pdev);
  1920.   int line_size_words = (line_size + W - 1) / W;
  1921.   int paper_size = gdev_pcl_paper_size((gx_device *)pdev);
  1922.   int num_comps = pdev->color_info.num_components;
  1923.   int bits_per_pixel = pdev->color_info.depth;
  1924.   int storage_bpp = bits_per_pixel;
  1925.   int expanded_bpp = bits_per_pixel;
  1926.   int plane_size, databuff_size;
  1927.   int combined_escapes = 1;
  1928.   int errbuff_size = 0;
  1929.   int outbuff_size = 0;
  1930.   int compression = 0;
  1931.   int scan = 0;
  1932.   int *errors[2];
  1933.   const char *cid_string = (const char*) 0;
  1934.   byte *data[4], *plane_data[4][4], *out_data;
  1935.   byte *out_row, *out_row_alt;
  1936.   word *storage;
  1937.   uint storage_size_words;
  1938.  
  1939.   /* Tricks and cheats ... */
  1940.   switch (ptype) {
  1941.   case DJ550C:
  1942.     if (num_comps == 3 && !cprn_device->cmyk)
  1943.       num_comps = 4;                      /* 4-component printing */
  1944.     break;
  1945.   case ESC_P:
  1946.     if (bits_per_pixel == 24)    /* prefer 3-component printing for bpp=24. */
  1947.       num_comps = 3;
  1948.     else
  1949.       if (num_comps != 1)
  1950.         num_comps = 4;
  1951.     break;
  1952.   case PJXL300:
  1953.   case PJXL180:
  1954.     if (pjxl->rendertype > 0) {
  1955.       if (bits_per_pixel < 16)
  1956.     pjxl->rendertype = 0;
  1957.       else {
  1958.     /* Control codes for CID sequence */
  1959.     cid_string = (bits_per_pixel == 16) ? CONFIG_16BIT : CONFIG_24BIT;
  1960.     /* Pretend we're a monobit device so we send the data out unchanged */
  1961.     bits_per_pixel = storage_bpp = expanded_bpp = 1;
  1962.     num_comps = 1;
  1963.       }
  1964.     }
  1965.     break;
  1966.   }
  1967.  
  1968.   if (cprn_device->cmyk <= 0) {
  1969.       if (storage_bpp == 8 && num_comps >= 3)
  1970.       bits_per_pixel = expanded_bpp = 3;    /* Only 3 bits of each byte used */
  1971.   }
  1972.  
  1973.   plane_size = calc_buffsize(line_size, storage_bpp);
  1974.   ep_plane_size = plane_size;
  1975.  
  1976.   if (bits_per_pixel == 1) {        /* Data printed direct from i/p */
  1977.     databuff_size = 0;            /* so no data buffer required, */
  1978.     outbuff_size = plane_size * 4;    /* but need separate output buffers */
  1979.   }
  1980.   
  1981.   if (bits_per_pixel > 4) {        /* Error buffer for FS dithering */
  1982.     storage_bpp = expanded_bpp = 
  1983.       num_comps * 8;            /* 8, 24 or 32 bits */
  1984.  
  1985.     if (cprn_device->cmyk > 0) {    /* Use CMYK dithering algorithm. */
  1986.     errbuff_size = 4 * (5 + 1 + 1 + line_size + 1 + 2) * I;
  1987.     } else {            /* Use original (RGB) dithering. */
  1988.     errbuff_size =            /* 4n extra values for line ends */
  1989.         calc_buffsize((plane_size * expanded_bpp + num_comps * 4) * I, 1);
  1990.     }
  1991.   }
  1992.  
  1993.   databuff_size = plane_size * storage_bpp;
  1994.  
  1995.   storage_size_words = ((plane_size + plane_size) * num_comps +
  1996.             databuff_size + errbuff_size + outbuff_size) / W;
  1997.  
  1998.   storage = (ulong *) gs_malloc(storage_size_words, W, "hp_colour_print_page");
  1999.   ep_storage_size_words = (plane_size * (num_comps + 1)) / W * img_rows
  2000.       + 16;            /* Redundant space for sentinel and aligning. */
  2001.   ep_storage = (word *) gs_malloc(ep_storage_size_words, W, "ep_print_buffer");
  2002.  
  2003.   /*
  2004.    * The principal data pointers are stored as pairs of values, with
  2005.    * the selection being made by the 'scan' variable. The function of the
  2006.    * scan variable is overloaded, as it controls both the alternating
  2007.    * raster scan direction used in the Floyd-Steinberg dithering and also
  2008.    * the buffer alternation required for line-difference compression.
  2009.    *
  2010.    * Thus, the number of pointers required is as follows:
  2011.    * 
  2012.    *   errors:      2  (scan direction only)
  2013.    *   data:        4  (scan direction and alternating buffers)
  2014.    *   plane_data:  4  (scan direction and alternating buffers)
  2015.    */
  2016.  
  2017.   if (storage == 0 || ep_storage == 0) /* can't allocate working area */
  2018.     return_error(gs_error_VMerror);
  2019.   else {
  2020.     int i, j;
  2021.     byte *p = out_data = out_row = (byte *)storage;
  2022.     byte *ep_p = (byte *)ep_storage;
  2023.     data[0] = data[1] = data[2] = p;
  2024.     data[3] = p + databuff_size;
  2025.     out_row_alt = out_row + plane_size * 2;
  2026.     if (bits_per_pixel > 1) {
  2027.       p += databuff_size;
  2028.     }
  2029.     if (bits_per_pixel > 4) {
  2030.       errors[0] = (int *)p + num_comps * 2;
  2031.       errors[1] = errors[0] + databuff_size;
  2032.       p += errbuff_size;
  2033.     }
  2034.     for (i = 0; i < num_comps; i++) {
  2035.       plane_data[0][i] = plane_data[2][i] = p;
  2036.       p += plane_size;
  2037.     }
  2038.     for (i = 0; i < num_comps; i++) {
  2039.       plane_data[1][i] = p;
  2040.       plane_data[3][i] = p + plane_size;
  2041.       p += plane_size;
  2042.     }
  2043.     if (bits_per_pixel == 1) {
  2044.       out_data = out_row = p;      /* size is outbuff_size * 4 */
  2045.       out_row_alt = out_row + plane_size * 2;
  2046.       data[1] += databuff_size;   /* coincides with plane_data pointers */
  2047.       data[3] += databuff_size;
  2048.     }
  2049.     for (i = 0; i < num_comps; i++) {
  2050.       for (j = 0; j < img_rows; j++) {
  2051.     ep_raster_buf[i][j] = ep_p;
  2052.     ep_p += plane_size;
  2053.       }
  2054.       /* Make a sentinel and align to word size.  */
  2055.       ep_print_buf = (byte *)((word)(ep_p + sizeof(word)) & ~(sizeof(word)-1));
  2056.     }
  2057.     ep_num_comps = num_comps;
  2058.   }
  2059.   
  2060.   /* Initialize printer. */
  2061.   if (ptype == BJC600 || ptype == BJC800) {
  2062.     bjc_init_page(pdev, prn_stream);
  2063.   } else {
  2064.       if (ptype == LJ4DITH)  {
  2065.       fputs("\033*rB", prn_stream);
  2066.       } else {
  2067.       fputs("\033*rbC", prn_stream);                   /* End raster graphics */
  2068.       }
  2069.       fprintf(prn_stream, "\033*t%dR", (int)x_dpi); 
  2070.   /* Set resolution */
  2071.   }
  2072.   
  2073.   /* Clear temp storage */
  2074.   memset(storage, 0, storage_size_words * W);
  2075.  
  2076. #define DOFFSET (dev_t_margin(pdev) - DESKJET_PRINT_LIMIT)    /* Print position */
  2077. #define POFFSET (dev_t_margin(pdev) - PAINTJET_PRINT_LIMIT)
  2078. #define EOFFSET (dev_t_margin(pdev) - ESC_P_PRINT_LIMIT)
  2079. #define BOFFSET (dev_t_margin(pdev) - bjc->printLimit)
  2080.   switch (ptype) {
  2081.   case LJ4DITH:
  2082.     /* Page size, orientation, top margin & perforation skip */
  2083.     fprintf(prn_stream, "\033&l26A\033&l0o0e0L\033*r0F" );
  2084.     fprintf(prn_stream, "\033*p0x0Y" ); /* These Offsets are hacked ! */
  2085.     fprintf(prn_stream, "\033&u600D\033*r1A" );
  2086.     /* Select data compression */
  2087.     compression = 3;
  2088.     combined_escapes = 0;
  2089.     break;
  2090.   case DJ500C:
  2091.   case DJ550C:
  2092.     /* Page size, orientation, top margin & perforation skip */
  2093.     fprintf(prn_stream, "\033&l%daolE", paper_size);
  2094.     /* Set depletion and shingling levels */
  2095.     fprintf(prn_stream, "\033*o%dd%dQ", cdj->depletion, cdj->shingling);
  2096.     /* Move to top left of printed area */
  2097.     fprintf(prn_stream, "\033*p%dY", (int)(300 * DOFFSET));
  2098.     /* Set number of planes ((-)1 is mono, (-)3 is (cmy)rgb, -4 is cmyk),
  2099.      * and raster width, then start raster graphics */
  2100.     fprintf(prn_stream, "\033*r%ds-%du0A", raster_width, num_comps);
  2101.     /* Select data compression */
  2102.     compression = 9;
  2103.     break;
  2104.   case DNJ650C:
  2105.     fprintf (prn_stream, "\033%%0B"); /* Enter HPGL/2 mode */
  2106.     fprintf (prn_stream, "BP5,1"); /* Turn off autorotation */
  2107.     fprintf (prn_stream, "PS%d,%d",
  2108.          (int)((pdev->height/pdev->y_pixels_per_inch)*1016),
  2109.          (int)((pdev->width/pdev->x_pixels_per_inch)*1016)); /* Set length/width of page */
  2110.     fprintf (prn_stream, "PU"); /* Pen up */
  2111.     fprintf (prn_stream, "PA%d,%d", 0, 0); /* Move pen to upper-left */
  2112.     fprintf (prn_stream, "\033%%1A"); /* Enter HP-RTL mode */
  2113.     fprintf (prn_stream, "\033&a1N"); /* No negative motion - allow plotting
  2114.                         while receiving */
  2115.     { static const char temp[] = {
  2116.         033, '*', 'v', '6', 'W',
  2117.     000 /* color model */,
  2118.     000 /* pixel encoding mode */,
  2119.     003 /* number of bits per index */,
  2120.     010 /* bits red */,
  2121.     010 /* bits green */,
  2122.     010 /* bits blue */
  2123.       };
  2124.       fwrite (temp, 1, sizeof(temp), prn_stream);
  2125.     }
  2126.  
  2127.     /* Set raster width */
  2128.     fprintf(prn_stream, "\033*r%dS", raster_width);
  2129.     /* Start raster graphics */
  2130.     fprintf(prn_stream, "\033*r1A");
  2131.  
  2132.     /* Select data compression */
  2133.     compression = 1;
  2134.     /* No combined escapes for raster transfers */
  2135.     combined_escapes = 0;
  2136.     break;
  2137.   case PJXL300:
  2138.     /* Page size, orientation, top margin & perforation skip */
  2139.     fprintf(prn_stream, "\033&l%daolE", paper_size);
  2140.     /* Set no-negative-motion mode, for faster (unbuffered) printing */
  2141.     fprintf(prn_stream, "\033&a1N");
  2142.     /* Set print quality */
  2143.     fprintf(prn_stream, "\033*o%dQ", pjxl->printqual);
  2144.     /* Move to top left of printed area */
  2145.     fprintf(prn_stream, "\033*p%dY", (int)(300 * POFFSET));
  2146.     /* Configure colour setup */
  2147.     if (pjxl->rendertype > 0) {
  2148.       /* Set render type */
  2149.       fprintf(prn_stream, "\033*t%dJ", pjxl->rendertype);
  2150.       /* Configure image data */
  2151.       fputs(cid_string, prn_stream);
  2152.       /* Set raster width, then start raster graphics */
  2153.       fprintf(prn_stream, "\033*r%ds1A", raster_width);
  2154.     } else {
  2155.       /* Set number of planes (1 is mono, 3 is rgb),
  2156.        * and raster width, then start raster graphics */
  2157.       fprintf(prn_stream, "\033*r%ds-%du0A", raster_width, num_comps);
  2158.     }
  2159.     /* No combined escapes for raster transfers */
  2160.     combined_escapes = 0;
  2161.     break;
  2162.   case PJXL180:
  2163.     /* Page size, orientation, top margin & perforation skip */
  2164.     fprintf(prn_stream, "\033&l%daolE", paper_size);
  2165.     /* Set print quality */
  2166.     fprintf(prn_stream, "\033*o%dQ", pjxl->printqual);
  2167.     /* Move to top left of printed area */
  2168.     fprintf(prn_stream, "\033*p%dY", (int)(180 * POFFSET));
  2169.     /* Configure colour setup */
  2170.     if (pjxl->rendertype > 0) {
  2171.       /* Set render type */
  2172.       fprintf(prn_stream, "\033*t%dJ", pjxl->rendertype);
  2173.       /* Configure image data */
  2174.       fputs(cid_string, prn_stream);
  2175.       /* Set raster width, then start raster graphics */
  2176.       fprintf(prn_stream, "\033*r%ds1A", raster_width);
  2177.     } else {
  2178.       /* Set number of planes (1 is mono, 3 is rgb),
  2179.        * and raster width, then start raster graphics */
  2180.       fprintf(prn_stream, "\033*r%ds%du0A", raster_width, num_comps);
  2181.     }
  2182.     break;
  2183.   case PJ180:
  2184.   case DECLJ250:
  2185.     /* Disable perforation skip */
  2186.     fprintf(prn_stream, "\033&lL");
  2187.     /* Move to top left of printed area */
  2188.     fprintf(prn_stream, "\033&a%dV", (int)(720 * POFFSET));
  2189.     /* Set number of planes (1 is mono, 3 is rgb),
  2190.      * and raster width, then start raster graphics */
  2191.     fprintf(prn_stream, "\033*r%ds%du0A", raster_width, num_comps);
  2192.     if (ptype == DECLJ250) {
  2193.       /* No combined escapes for raster transfers */
  2194.       combined_escapes = 0;
  2195.       /* From here on, we're a standard Paintjet .. */
  2196.       ptype = PJ180;
  2197.     }
  2198.     /* Select data compression */
  2199.     compression = 1;
  2200.     break;
  2201.   case ESC_P:
  2202.     /* Move to top left of printed area (must be modified for large movement(YK))*/
  2203.     if ((int)(EOFFSET*360)) fprintf(prn_stream, "\033|J%c%c", 0, (int)(360*EOFFSET));
  2204.     combined_escapes = 0;
  2205.     break;
  2206.   case BJC600:
  2207.   case BJC800:
  2208.     /* Move to top left of printed area */
  2209.     bjc_v_skip((int)(pdev->HWResolution[1] * BOFFSET), pdev, prn_stream);
  2210.     combined_escapes = 0;
  2211.     compression = 2;        /* BJC600 uses the same method as mode 2 compression */
  2212.     break;
  2213.   }
  2214.  
  2215.   /* Unfortunately, the Paintjet XL300 PCL interpreter introduces a
  2216.    * version of the PCL language which is different to all earlier HP
  2217.    * colour and mono inkjets, in that it loses the very useful ability
  2218.    * to use combined escape sequences with the raster transfer
  2219.    * commands. In this respect, it is incompatible even with the older
  2220.    * 180 dpi PaintJet and PaintJet XL printers!  Another regrettable
  2221.    * omission is that 'mode 9' compression is not supported, as this
  2222.    * mode can give both computational and PCL file size advantages. */
  2223.  
  2224.   if (combined_escapes) {
  2225.     /* From now on, all escape commands start with \033*b, so we
  2226.      * combine them (if the printer supports this). */
  2227.     fputs("\033*b", prn_stream);
  2228.      /* Set compression if the mode has been defined. */
  2229.     if (compression)
  2230.       fprintf(prn_stream, "%dm", compression);
  2231.   }
  2232.   else if (ptype == BJC600 || ptype == BJC800)
  2233.       ;                /* Currently, nothing to do. */
  2234.   else
  2235.     if (compression)
  2236.       fprintf(prn_stream, "\033*b%dM", compression);
  2237.  
  2238.   /* Send each scan line in turn */
  2239.   {
  2240.     int cErr, mErr, yErr, kErr;
  2241.     int this_pass, lnum, i;
  2242.     int start_rows;
  2243.     int lend, num_blank_lines = 0;
  2244.  
  2245.     word rmask = ~(word) 0 << ((-pdev->width * storage_bpp) & (W * 8 - 1));
  2246.  
  2247.     lend = pdev->height - (dev_t_margin(pdev) + dev_b_margin(pdev)) * y_dpi;
  2248.  
  2249.     switch (ptype) {
  2250.     case BJC600:
  2251.     case BJC800:
  2252.         start_rows = BJC_HEAD_ROWS;
  2253.         break;
  2254.  
  2255.     /* Inhibit blank line printing for RGB-only printers, since in
  2256.      * this case 'blank' means black!  Also disabled for XL300 due to
  2257.      * an obscure bug in the printer's firmware */
  2258.  
  2259.     case PJ180:
  2260.     case PJXL180:
  2261.     case PJXL300:
  2262.         start_rows = -1;
  2263.         break;
  2264.  
  2265.     default:
  2266.         start_rows = (num_comps == 1) ? HEAD_ROWS_MONO - 1 :
  2267.         HEAD_ROWS_COLOUR - 1;
  2268.         break;
  2269.     }
  2270.  
  2271.     cErr = mErr = yErr = kErr = 0;
  2272.  
  2273.     if (bits_per_pixel > 4) { /* Randomly seed initial error buffer */
  2274.       if (cprn_device->cmyk > 0 && expanded_bpp == 32) {
  2275.       bjc_fscmyk(data, plane_data, errors, plane_size, -1);
  2276.       } else {
  2277.       int *ep = errors[0];
  2278.       for (i = 0; i < databuff_size; i++) {
  2279.           *ep++ = RANDOM;
  2280.       }
  2281.       }
  2282.     }
  2283.  
  2284.     this_pass = start_rows;
  2285.     for (lnum = 0; lnum < lend; lnum++) {
  2286.       word *data_words = (word *)data[scan];
  2287.       register word *end_data = data_words + line_size_words;
  2288.  
  2289.       gdev_prn_copy_scan_lines(pdev, lnum, data[scan], line_size);
  2290.  
  2291.       /* Mask off 1-bits beyond the line width. */
  2292.       end_data[-1] &= rmask;
  2293.  
  2294.       /* Remove trailing 0s. */
  2295.       while (end_data > data_words && end_data[-1] == 0)
  2296.     end_data--;
  2297.       if (ptype != DNJ650C)    /* DesignJet can't skip blank lines ? ? */
  2298.     if (end_data == data_words) {    /* Blank line */
  2299.       num_blank_lines++;
  2300.       continue;
  2301.     }
  2302.       /* Skip blank lines if any */
  2303.       if (num_blank_lines > 0) {
  2304.     if (ptype == ESC_P) {
  2305.       ep_print_image(prn_stream, 'B', 0, num_blank_lines);
  2306.     } else if (ptype == BJC600 || ptype == BJC800) {
  2307.         bjc_v_skip(num_blank_lines, pdev, prn_stream);
  2308.     } else if (num_blank_lines < this_pass) {
  2309.       /* Moving down from current position
  2310.        * causes head motion on the DeskJets, so
  2311.        * if the number of lines is within the
  2312.        * current pass of the print head, we're
  2313.        * better off printing blanks. */
  2314.       this_pass -= num_blank_lines;
  2315.       if (combined_escapes) {
  2316.         fputc('y', prn_stream);   /* Clear current and seed rows */
  2317.         for (; num_blank_lines; num_blank_lines--)
  2318.           fputc('w', prn_stream);
  2319.       } else {
  2320. #if 0
  2321. /**************** The following code has been proposed ****************/
  2322. /**************** as a replacement: ****************/
  2323.         fputs("\033*b1Y", prn_stream);   /* Clear current and seed rows */
  2324.         if ( num_blank_lines > 1 )
  2325.           fprintf(prn_stream, "\033*b%dY", num_blank_lines - 1);
  2326.         num_blank_lines = 0;
  2327. #else
  2328.         fputs("\033*bY", prn_stream);   /* Clear current and seed rows */
  2329.         if (ptype == DNJ650C) {
  2330.           fprintf (prn_stream, "\033*b%dY", num_blank_lines);
  2331.           num_blank_lines = 0;
  2332.         }
  2333.         else {
  2334.           for (; num_blank_lines; num_blank_lines--)
  2335.             fputs("\033*bW", prn_stream);
  2336.         }
  2337. #endif
  2338.       }
  2339.     } else {
  2340.       if (combined_escapes)
  2341.         fprintf(prn_stream, "%dy", num_blank_lines);
  2342.       else
  2343.         fprintf(prn_stream, "\033*b%dY", num_blank_lines);
  2344.     }
  2345.     memset(plane_data[1 - scan][0], 0, plane_size * num_comps);
  2346.     num_blank_lines = 0;
  2347.     this_pass = start_rows;
  2348.       }
  2349.       {            /* Printing non-blank lines */
  2350.     register byte *kP = plane_data[scan + 2][3];
  2351.     register byte *cP = plane_data[scan + 2][2];
  2352.     register byte *mP = plane_data[scan + 2][1];
  2353.     register byte *yP = plane_data[scan + 2][0];
  2354.     register byte *dp = data[scan + 2];
  2355.     register int *ep = errors[scan];
  2356.     int zero_row_count;
  2357.     int i, j;
  2358.     byte *odp;
  2359.  
  2360.     if (this_pass)
  2361.       this_pass--;
  2362.     else
  2363.       this_pass = start_rows;
  2364.  
  2365.     if (expanded_bpp > bits_per_pixel) {  /* Expand line if required */
  2366.         cdj_expand_line(data_words, line_size,
  2367.         cprn_device->cmyk,
  2368.             bits_per_pixel, expanded_bpp);
  2369.     }
  2370.  
  2371.     /* In colour modes, we have some bit-shuffling to do before
  2372.      * we can print the data; in FS mode we also have the
  2373.      * dithering to take care of. */
  2374.     switch (expanded_bpp) {    /* Can be 1, 3, 8, 24 or 32 */
  2375.     case 3:
  2376.       /* Transpose the data to get pixel planes. */
  2377.       for (i = 0, odp = plane_data[scan][0]; i < databuff_size;
  2378.            i += 8, odp++) {    /* The following is for 16-bit
  2379.                  * machines */
  2380. #define spread3(c)\
  2381.     { 0, c, c*0x100, c*0x101, c*0x10000L, c*0x10001L, c*0x10100L, c*0x10101L }
  2382.         static ulong spr40[8] = spread3(0x40);
  2383.         static ulong spr08[8] = spread3(8);
  2384.         static ulong spr02[8] = spread3(2);
  2385.         register byte *dp = data[scan] + i;
  2386.         register ulong pword =
  2387.         (spr40[dp[0]] << 1) +
  2388.         (spr40[dp[1]]) +
  2389.         (spr40[dp[2]] >> 1) +
  2390.         (spr08[dp[3]] << 1) +
  2391.         (spr08[dp[4]]) +
  2392.         (spr08[dp[5]] >> 1) +
  2393.         (spr02[dp[6]]) +
  2394.         (spr02[dp[7]] >> 1);
  2395.         odp[0] = (byte) (pword >> 16);
  2396.         odp[plane_size] = (byte) (pword >> 8);
  2397.         odp[plane_size * 2] = (byte) (pword);
  2398.       }
  2399.       break;
  2400.  
  2401.     case 8:
  2402.       switch (ptype) {
  2403.           case BJC600:
  2404.           case BJC800:
  2405.               if (bjcparams.ditheringType == BJC_DITHER_NONE) {
  2406.               COPYline(scan, i, j, plane_size, cP, mP, yP, kP, 1);
  2407.               break;
  2408.           }
  2409.  
  2410.           default:
  2411.           FSDline(scan, i, j, plane_size, cErr, mErr, yErr, kErr,
  2412.               cP, mP, yP, kP, 1);
  2413.       }
  2414.       break;
  2415.     case 24:
  2416.       FSDline(scan, i, j, plane_size, cErr, mErr, yErr, kErr,
  2417.           cP, mP, yP, kP, 3);
  2418.       break;
  2419.     case 32:
  2420.       if (cprn_device->cmyk > 0) {
  2421.           bjc_fscmyk(data, plane_data, errors, plane_size, scan);
  2422.       } else {
  2423.           FSDline(scan, i, j, plane_size, cErr, mErr, yErr, kErr,
  2424.               cP, mP, yP, kP, 4);
  2425.       }
  2426.       break;
  2427.  
  2428.     } /* switch(expanded_bpp) */
  2429.  
  2430.     /* Make sure all black is in the k plane */
  2431.  
  2432.     if (num_comps == 4 && (cprn_device->cmyk <= 0 || expanded_bpp != 32)) {
  2433.         register word *kp = (word *)plane_data[scan][3];
  2434.         register word *cp = (word *)plane_data[scan][2];
  2435.         register word *mp = (word *)plane_data[scan][1];
  2436.         register word *yp = (word *)plane_data[scan][0];
  2437.         if (bits_per_pixel > 4) {  /* Done as 4 planes */
  2438.         for (i = 0; i < plane_size / W; i++) {
  2439.             word bits = *cp & *mp & *yp;
  2440.             *kp++ |= bits;
  2441.              bits = ~bits;
  2442.             *cp++ &= bits;
  2443.             *mp++ &= bits;
  2444.             *yp++ &= bits;
  2445.         }
  2446.         } else {  /* This has really been done as 3 planes */
  2447.         for (i = 0; i < plane_size / W; i++) {
  2448.             word bits = *cp & *mp & *yp;
  2449.             *kp++ = bits;
  2450.             bits = ~bits;
  2451.             *cp++ &= bits;
  2452.             *mp++ &= bits;
  2453.             *yp++ &= bits;
  2454.         }
  2455.         }
  2456.     }
  2457.  
  2458.     /* Transfer raster graphics in the order (K), C, M, Y */
  2459.  
  2460.     for (zero_row_count = 0, i = num_comps - 1; i >= 0; i--) {
  2461.       int output_plane = 1;
  2462.       int out_count = 0;
  2463.       
  2464.       switch (ptype) {
  2465.       case DJ500C:    /* Always compress using mode 9 */
  2466.       case DJ550C:
  2467.         out_count = gdev_pcl_mode9compress(plane_size,
  2468.                            plane_data[scan][i],
  2469.                            plane_data[1 - scan][i],
  2470.                            out_data);
  2471.  
  2472.         /* This optimisation allows early termination of the
  2473.          * row, but this doesn't work correctly in an alternating
  2474.          * mode 2 / mode 3 regime, so we only use it with mode 9
  2475.          * compression */
  2476.            if (out_count == 0)
  2477.              { output_plane = 0;      /* No further output for this plane */
  2478.                if (i == 0)
  2479.                  fputc('w', prn_stream);
  2480.                else
  2481.                  zero_row_count++;
  2482.              }
  2483.            else
  2484.              { for (; zero_row_count; zero_row_count--)
  2485.                  fputc('v', prn_stream);
  2486.              }
  2487.         break;
  2488.       case PJ180:
  2489.       case DNJ650C:
  2490.         if (num_comps > 1)
  2491.           { word *wp = (word *)plane_data[scan][i];
  2492.         for (j = 0; j < plane_size / W; j++, wp++)
  2493.           *wp = ~*wp;
  2494.           }
  2495.         out_count = gdev_pcl_mode1compress((const byte *)
  2496.                            plane_data[scan][i],
  2497.                            (const byte *)
  2498.                            plane_data[scan][i] + plane_size - 1,
  2499.                            out_data);
  2500.         break;
  2501.       case PJXL180:    /* Need to invert data as CMY not supported */
  2502.         if (num_comps > 1)
  2503.           { word *wp = (word *)plane_data[scan][i];
  2504.         for (j = 0; j < plane_size / W; j++, wp++)
  2505.           *wp = ~*wp;
  2506.           }
  2507.         /* fall through .. */
  2508.       case PJXL300:     /* Compression modes 2 and 3 are both 
  2509.                  * available.  Try both and see which one
  2510.                  * produces the least output data. */
  2511.       case LJ4DITH:
  2512.         { const byte *plane = plane_data[scan][i];
  2513.           byte *prev_plane = plane_data[1 - scan][i];
  2514.           const word *row = (word *)plane;
  2515.           const word *end_row = row + plane_size/W;
  2516.           int count2 = gdev_pcl_mode2compress(row, end_row, out_row_alt);
  2517.           int count3 = gdev_pcl_mode3compress(plane_size, plane, prev_plane, out_row);
  2518.           int penalty = combined_escapes ? strlen("#m") : strlen("\033*b#M");
  2519.           int penalty2 = (compression == 2 ? 0 : penalty);
  2520.           int penalty3 = (compression == 3 ? 0 : penalty);
  2521.           
  2522.           if (count3 + penalty3 < count2 + penalty2)
  2523.         { if ( compression != 3 ) {
  2524.             if (combined_escapes)
  2525.               fputs("3m", prn_stream);
  2526.             else
  2527.               fputs("\033*b3M", prn_stream);
  2528.             compression = 3;
  2529.           }
  2530.           out_data = out_row;
  2531.           out_count = count3;
  2532.         }
  2533.           else
  2534.         { if ( compression != 2 ) {
  2535.             if (combined_escapes)
  2536.               fputs("2m", prn_stream);
  2537.             else
  2538.               fputs("\033*b2M", prn_stream);
  2539.             compression = 2;
  2540.           }
  2541.           out_data = out_row_alt;
  2542.           out_count = count2;
  2543.         }
  2544.         }
  2545.         break;
  2546.       case BJC600:
  2547.       case BJC800:
  2548.         { const byte *plane = (byte *)plane_data[scan][i];
  2549.           int count2 = bjc_compress(plane, plane + plane_size, out_row_alt);
  2550.  
  2551.            out_data = out_row_alt;
  2552.           out_count = count2;
  2553.         }
  2554.         break;
  2555.       }
  2556.       if (output_plane) {
  2557.         if (combined_escapes)
  2558.           fprintf(prn_stream, "%d%c", out_count, "wvvv"[i]);
  2559.         else if (ptype == BJC600 || ptype == BJC800) {
  2560.           if (out_count)
  2561.           bjc_raster_cmd(num_comps == 1 ? 3 : i,
  2562.                 out_count, out_data, pdev, prn_stream);
  2563.           if (i == 0) bjc_v_skip(1, pdev, prn_stream);
  2564.         } else if (ptype == ESC_P)
  2565.         ep_print_image(prn_stream, i, plane_data[scan][i], plane_size);
  2566.         else
  2567.           fprintf(prn_stream, "\033*b%d%c", out_count, "WVVV"[i]);
  2568.         if (ptype < ESC_P)
  2569.           fwrite(out_data, sizeof(byte), out_count, prn_stream);
  2570.       }
  2571.       
  2572.     } /* Transfer Raster Graphics ... */
  2573.     if (ptype == ESC_P)
  2574.         ep_print_image(prn_stream, 'I', 0, 0); /* increment line index */
  2575.     scan = 1 - scan;          /* toggle scan direction */
  2576.       }      /* Printing non-blank lines */
  2577.     }     /* for lnum ... */
  2578.   }       /* send each scan line in turn */
  2579.  
  2580.   if (combined_escapes)
  2581.     fputs("0M", prn_stream);
  2582.  
  2583.   /* end raster graphics */
  2584.   if (ptype == BJC600 || ptype == BJC800) {
  2585.     bjc_finish_page(pdev, prn_stream);
  2586.   }
  2587.   else if (ptype != ESC_P) 
  2588.     fputs("\033*rbC\033E", prn_stream);
  2589.  
  2590.   /* eject page */
  2591.   if (ptype == PJ180)
  2592.     fputc('\f', prn_stream);
  2593.   else if (ptype == DNJ650C)
  2594.     fputs ("\033*rC\033%%0BPG;", prn_stream);
  2595.   else if (ptype == BJC600 || ptype == BJC800)
  2596.       ;                /* Already done */
  2597.   else if (ptype == ESC_P) {
  2598.     ep_print_image(prn_stream, 'F', 0, 0); /* flush print buffer */
  2599.     fputs("\014\033@", prn_stream);    /* reset after eject page */
  2600.   } else 
  2601.     fputs("\033&l0H", prn_stream);
  2602.  
  2603.   /* free temporary storage */
  2604.   gs_free((char *) ep_storage, ep_storage_size_words, W, "ep_print_buffer");
  2605.   gs_free((char *) storage, storage_size_words, W, "hp_colour_print_page");
  2606.  
  2607.   return 0;
  2608. }
  2609.  
  2610. /*
  2611.  * Mode 9 2D compression for the HP DeskJet 5xxC. This mode can give
  2612.  * very good compression ratios, especially if there are areas of flat
  2613.  * colour (or blank areas), and so is 'highly recommended' for colour
  2614.  * printing in particular because of the very large amounts of data which
  2615.  * can be generated
  2616.  */
  2617. private int
  2618. gdev_pcl_mode9compress(int bytecount, const byte * current, const byte * previous, byte * compressed)
  2619. {
  2620.   register const byte *cur = current;
  2621.   register const byte *prev = previous;
  2622.   register byte *out = compressed;
  2623.   const byte *end = current + bytecount;
  2624.  
  2625.   while (cur < end) {        /* Detect a run of unchanged bytes. */
  2626.     const byte *run = cur;
  2627.     register const byte *diff;
  2628.     int offset;
  2629.     while (cur < end && *cur == *prev) {
  2630.       cur++, prev++;
  2631.     }
  2632.     if (cur == end)
  2633.       break;            /* rest of row is unchanged */
  2634.     /* Detect a run of changed bytes. */
  2635.     /* We know that *cur != *prev. */
  2636.     diff = cur;
  2637.     do {
  2638.       prev++;
  2639.       cur++;
  2640.     }
  2641.     while (cur < end && *cur != *prev);
  2642.     /* Now [run..diff) are unchanged, and */
  2643.     /* [diff..cur) are changed. */
  2644.     offset = diff - run;
  2645.     {
  2646.       const byte *stop_test = cur - 4;
  2647.       int dissimilar, similar;
  2648.  
  2649.       while (diff < cur) {
  2650.     const byte *compr = diff;
  2651.     const byte *next;    /* end of run */
  2652.     byte value = 0;
  2653.     while (diff <= stop_test &&
  2654.            ((value = *diff) != diff[1] ||
  2655.         value != diff[2] ||
  2656.         value != diff[3]))
  2657.       diff++;
  2658.  
  2659.     /* Find out how long the run is */
  2660.     if (diff > stop_test)    /* no run */
  2661.       next = diff = cur;
  2662.     else {
  2663.       next = diff + 4;
  2664.       while (next < cur && *next == value)
  2665.         next++;
  2666.     }
  2667.  
  2668. #define MAXOFFSETU 15
  2669. #define MAXCOUNTU 7
  2670.     /* output 'dissimilar' bytes, uncompressed */
  2671.     if ((dissimilar = diff - compr)) {
  2672.       int temp, i;
  2673.  
  2674.       if ((temp = --dissimilar) > MAXCOUNTU)
  2675.         temp = MAXCOUNTU;
  2676.       if (offset < MAXOFFSETU)
  2677.         *out++ = (offset << 3) | (byte) temp;
  2678.       else {
  2679.         *out++ = (MAXOFFSETU << 3) | (byte) temp;
  2680.         offset -= MAXOFFSETU;
  2681.         while (offset >= 255) {
  2682.           *out++ = 255;
  2683.           offset -= 255;
  2684.         }
  2685.         *out++ = offset;
  2686.       }
  2687.       if (temp == MAXCOUNTU) {
  2688.         temp = dissimilar - MAXCOUNTU;
  2689.         while (temp >= 255) {
  2690.           *out++ = 255;
  2691.           temp -= 255;
  2692.         }
  2693.         *out++ = (byte) temp;
  2694.       }
  2695.       for (i = 0; i <= dissimilar; i++)
  2696.         *out++ = *compr++;
  2697.       offset = 0;
  2698.     }            /* end uncompressed */
  2699. #define MAXOFFSETC 3
  2700. #define MAXCOUNTC 31
  2701.     /* output 'similar' bytes, run-length encoded */
  2702.     if ((similar = next - diff)) {
  2703.       int temp;
  2704.  
  2705.       if ((temp = (similar -= 2)) > MAXCOUNTC)
  2706.         temp = MAXCOUNTC;
  2707.       if (offset < MAXOFFSETC)
  2708.         *out++ = 0x80 | (offset << 5) | (byte) temp;
  2709.       else {
  2710.         *out++ = 0x80 | (MAXOFFSETC << 5) | (byte) temp;
  2711.         offset -= MAXOFFSETC;
  2712.         while (offset >= 255) {
  2713.           *out++ = 255;
  2714.           offset -= 255;
  2715.         }
  2716.         *out++ = offset;
  2717.       }
  2718.       if (temp == MAXCOUNTC) {
  2719.         temp = similar - MAXCOUNTC;
  2720.         while (temp >= 255) {
  2721.           *out++ = 255;
  2722.           temp -= 255;
  2723.         }
  2724.         *out++ = (byte) temp;
  2725.       }
  2726.       *out++ = value;
  2727.       offset = 0;
  2728.     }            /* end compressed */
  2729.     diff = next;
  2730.       }
  2731.     }
  2732.   }
  2733.   return out - compressed;
  2734. }
  2735.  
  2736. /*
  2737.  * Row compression for the H-P PaintJet.
  2738.  * Compresses data from row up to end_row, storing the result
  2739.  * starting at compressed.  Returns the number of bytes stored.
  2740.  * The compressed format consists of a byte N followed by a
  2741.  * data byte that is to be repeated N+1 times.
  2742.  * In the worst case, the `compressed' representation is
  2743.  * twice as large as the input.
  2744.  * We complement the bytes at the same time, because
  2745.  * we accumulated the image in complemented form.
  2746.  */
  2747. private int
  2748. gdev_pcl_mode1compress(const byte *row, const byte *end_row, byte *compressed)
  2749. {    register const byte *in = row;
  2750.     register byte *out = compressed;
  2751.     while ( in < end_row )
  2752.            {    byte test = *in++;
  2753.         const byte *run = in;
  2754.         while ( in < end_row && *in == test ) in++;
  2755.         /* Note that in - run + 1 is the repetition count. */
  2756.         while ( in - run > 255 )
  2757.                    {    *out++ = 255;
  2758.             *out++ = test;
  2759.             run += 256;
  2760.                    }
  2761.         *out++ = in - run;
  2762.         *out++ = test;
  2763.            }
  2764.     return out - compressed;
  2765. }
  2766.  
  2767. /*
  2768.  * Map a CMYK color to a color index. We just use depth / 4 bits per color
  2769.  * to produce the color index.
  2770.  *
  2771.  * Important note: CMYK values are stored in the order K, C, M, Y because of
  2772.  * the way the HP drivers work.
  2773.  *
  2774.  */
  2775.  
  2776. #define gx_color_value_to_bits(cv, b) \
  2777.     ((cv) >> (gx_color_value_bits - (b)))
  2778. #define gx_bits_to_color_value(cv, b) \
  2779.     ((cv) << (gx_color_value_bits - (b)))
  2780.  
  2781. #define gx_cmyk_value_bits(c, m, y, k, b) \
  2782.     ((gx_color_value_to_bits((k), (b)) << (3 * (b))) | \
  2783.      (gx_color_value_to_bits((c), (b)) << (2 * (b))) | \
  2784.      (gx_color_value_to_bits((m), (b)) << (b)) | \
  2785.      (gx_color_value_to_bits((y), (b))))
  2786.  
  2787. #define gx_value_cmyk_bits(v, c, m, y, k, b) \
  2788.     (k) = gx_bits_to_color_value(((v) >> (3 * (b))) & ((1 << (b)) - 1), (b)), \
  2789.     (c) = gx_bits_to_color_value(((v) >> (2 * (b))) & ((1 << (b)) - 1), (b)), \
  2790.     (m) = gx_bits_to_color_value(((v) >> (b)) & ((1 << (b)) - 1), (b)), \
  2791.     (y) = gx_bits_to_color_value((v) & ((1 << (b)) - 1), (b))
  2792.  
  2793. private gx_color_index
  2794. gdev_cmyk_map_cmyk_color(gx_device* pdev,
  2795.     gx_color_value cyan, gx_color_value magenta, gx_color_value yellow,
  2796.     gx_color_value black) {
  2797.  
  2798.     gx_color_index color;
  2799.  
  2800.     switch (pdev->color_info.depth) {
  2801.     case 1:
  2802.        color = (cyan | magenta | yellow | black) > gx_max_color_value / 2 ?
  2803.            (gx_color_index) 1 : (gx_color_index) 0;
  2804.        break;
  2805.  
  2806.     default: {
  2807.         int nbits = pdev->color_info.depth;
  2808.  
  2809.             if (cyan == magenta && magenta == yellow) {
  2810.  
  2811.             /* Convert CMYK to gray -- Red Book 6.2.2 */
  2812.  
  2813.             float bpart = ((float) cyan) * (lum_red_weight / 100.) +
  2814.             ((float) magenta) * (lum_green_weight / 100.) +
  2815.             ((float) yellow) * (lum_blue_weight / 100.) +
  2816.             (float) black;
  2817.  
  2818.         cyan = magenta = yellow = (gx_color_index) 0;
  2819.         black = (gx_color_index) (bpart > gx_max_color_value ?
  2820.             gx_max_color_value : bpart);
  2821.         }
  2822.  
  2823.         color = gx_cmyk_value_bits(cyan, magenta, yellow, black,
  2824.             nbits >> 2);
  2825.      }
  2826.    }
  2827.  
  2828.    return color;
  2829. }
  2830.  
  2831. /* Mapping of RGB colors to gray values. */
  2832.  
  2833. private gx_color_index
  2834. gdev_cmyk_map_rgb_color(gx_device *pdev, gx_color_value r, gx_color_value g, gx_color_value b)
  2835. {
  2836.  
  2837.   if (gx_color_value_to_byte(r & g & b) == 0xff) {
  2838.       return (gx_color_index) 0;    /* White */
  2839.   } else {
  2840.       gx_color_value c = gx_max_color_value - r;
  2841.       gx_color_value m = gx_max_color_value - g;
  2842.       gx_color_value y = gx_max_color_value - b;
  2843.  
  2844.       switch (pdev->color_info.depth) {
  2845.       case 1:
  2846.           return (c | m | y) > gx_max_color_value / 2 ?
  2847.               (gx_color_index) 1 : (gx_color_index) 0;
  2848.           /*NOTREACHED*/
  2849.           break;
  2850.  
  2851.       case 8:
  2852.           return ((ulong) c * lum_red_weight * 10
  2853.               + (ulong) m * lum_green_weight * 10
  2854.               + (ulong) y * lum_blue_weight * 10)
  2855.           >> (gx_color_value_bits + 2);
  2856.           /*NOTREACHED*/
  2857.           break;
  2858.       }
  2859.   }
  2860.  
  2861.    return (gx_color_index) 0;    /* This should never happen. */
  2862. }
  2863.  
  2864. /* Mapping of CMYK colors. */
  2865.  
  2866. private int
  2867. gdev_cmyk_map_color_rgb(gx_device *pdev, gx_color_index color, gx_color_value prgb[3])
  2868. {
  2869.     switch (pdev->color_info.depth) {
  2870.     case 1:
  2871.        prgb[0] = prgb[1] = prgb[2] = gx_max_color_value * (1 - color);
  2872.        break;
  2873.  
  2874.     case 8:
  2875.        if (pdev->color_info.num_components == 1) {
  2876.            gx_color_value value = (gx_color_value) color ^ 0xff;
  2877.  
  2878.            prgb[0] = prgb[1] = prgb[2] = (value << 8) + value;
  2879.  
  2880.            break;
  2881.        }
  2882.  
  2883.     default: {
  2884.         unsigned long bcyan, bmagenta, byellow, black;
  2885.         int nbits = pdev->color_info.depth;
  2886.  
  2887.         gx_value_cmyk_bits(color, bcyan, bmagenta, byellow, black,
  2888.             nbits >> 2);
  2889.  
  2890. #ifdef USE_ADOBE_CMYK_RGB
  2891.  
  2892.         /* R = 1.0 - min(1.0, C + K), etc. */
  2893.  
  2894.         bcyan += black, bmagenta += black, byellow += black;
  2895.         prgb[0] = (bcyan > gx_max_color_value ? (gx_color_value) 0 :
  2896.                gx_max_color_value - bcyan);
  2897.         prgb[1] = (bmagenta > gx_max_color_value ? (gx_color_value) 0 :
  2898.                gx_max_color_value - bmagenta);
  2899.         prgb[2] = (byellow > gx_max_color_value ? (gx_color_value) 0 :
  2900.                gx_max_color_value - byellow);
  2901.  
  2902. #else
  2903.  
  2904.         /* R = (1.0 - C) * (1.0 - K), etc. */
  2905.  
  2906.         prgb[0] = (gx_color_value)
  2907.           ((ulong)(gx_max_color_value - bcyan) *
  2908.            (gx_max_color_value - black) / gx_max_color_value);
  2909.         prgb[1] = (gx_color_value)
  2910.           ((ulong)(gx_max_color_value - bmagenta) *
  2911.            (gx_max_color_value - black) / gx_max_color_value);
  2912.         prgb[2] = (gx_color_value)
  2913.           ((ulong)(gx_max_color_value - byellow) *
  2914.            (gx_max_color_value - black) / gx_max_color_value);
  2915.  
  2916. #endif
  2917.  
  2918.     }
  2919.     }
  2920.  
  2921.     return 0;
  2922. }
  2923.  
  2924. /*
  2925.  * Map a r-g-b color to a color index.
  2926.  * We complement the colours, since we're using cmy anyway, and
  2927.  * because the buffering routines expect white to be zero.
  2928.  * Includes colour balancing, following HP recommendations, to try
  2929.  * and correct the greenish cast resulting from an equal mix of the
  2930.  * c, m, y, inks by reducing the cyan component to give a truer black.
  2931.  */
  2932.  
  2933. /* Simple black generation/under-color removal with BG(k) = UG(k) = k. YA. */
  2934.  
  2935. #define bg_and_ucr(c, c_v, m, m_v, y, y_v, k) \
  2936.     do { \
  2937.        register byte cv = c_v, mv = m_v, yv = y_v, kv; \
  2938.  \
  2939.         kv = (cv > mv ? mv : cv); \
  2940.     kv = (yv > k ? k : y); \
  2941.         y = yv - kv; m = mv - kv; c = cv -kv; k = kv; \
  2942.    } while (0)
  2943.  
  2944. private gx_color_index
  2945. gdev_pcl_map_rgb_color(gx_device *pdev, gx_color_value r,
  2946.                  gx_color_value g, gx_color_value b)
  2947. {
  2948.   if (gx_color_value_to_byte(r & g & b) == 0xff)
  2949.     return (gx_color_index)0;         /* white */
  2950.   else {
  2951.     int correction = cprn_device->correction;
  2952.     gx_color_value c = gx_max_color_value - r;
  2953.     gx_color_value m = gx_max_color_value - g;
  2954.     gx_color_value y = gx_max_color_value - b;
  2955.     
  2956.     /* Colour correction for better blacks when using the colour ink
  2957.      * cartridge (on the DeskJet 500C only). We reduce the cyan component
  2958.      * by some fraction (eg. 4/5) to correct the slightly greenish cast
  2959.      * resulting from an equal mix of the three inks */
  2960.     if (correction) {
  2961.       ulong maxval, minval, range;
  2962.       
  2963.       maxval = c >= m ? (c >= y ? c : y) : (m >= y ? m : y);
  2964.       if (maxval > 0) {
  2965.     minval = c <= m ? (c <= y ? c : y) : (m <= y? m : y);
  2966.     range = maxval - minval;
  2967.     
  2968. #define shift (gx_color_value_bits - 12)
  2969.     c = ((c >> shift) * (range + (maxval * correction))) /
  2970.       ((maxval * (correction + 1)) >> shift);
  2971.       }
  2972.     }
  2973.     
  2974.     switch (pdev->color_info.depth) {
  2975.     case 1:
  2976.       return ((c | m | y) > gx_max_color_value / 2 ?
  2977.           (gx_color_index)1 : (gx_color_index)0);
  2978.     case 8:
  2979.       if (pdev->color_info.num_components >= 3)
  2980. #define gx_color_value_to_1bit(cv) ((cv) >> (gx_color_value_bits - 1))
  2981.     return (gx_color_value_to_1bit(c) +
  2982.         (gx_color_value_to_1bit(m) << 1) +
  2983.         (gx_color_value_to_1bit(y) << 2));
  2984.       else
  2985. #define red_weight 306
  2986. #define green_weight 601
  2987. #define blue_weight 117
  2988.     return ((((ulong)c * red_weight +
  2989.           (ulong)m * green_weight +
  2990.           (ulong)y * blue_weight)
  2991.          >> (gx_color_value_bits + 2)));
  2992.     case 16:
  2993. #define gx_color_value_to_5bits(cv) ((cv) >> (gx_color_value_bits - 5))
  2994. #define gx_color_value_to_6bits(cv) ((cv) >> (gx_color_value_bits - 6))
  2995.       return (gx_color_value_to_5bits(y) +
  2996.           (gx_color_value_to_6bits(m) << 5) +
  2997.           (gx_color_value_to_5bits(c) << 11));
  2998.     case 24:
  2999.       return (gx_color_value_to_byte(y) +
  3000.           (gx_color_value_to_byte(m) << 8) +
  3001.           ((ulong)gx_color_value_to_byte(c) << 16));
  3002.     case 32:
  3003.       { return ((c == m && c == y) ? ((ulong)gx_color_value_to_byte(c) << 24)
  3004.      : (gx_color_value_to_byte(y) +
  3005.         (gx_color_value_to_byte(m) << 8) +
  3006.         ((ulong)gx_color_value_to_byte(c) << 16)));
  3007.       }
  3008.     }
  3009.   }
  3010.   return (gx_color_index)0;   /* This never happens */
  3011. }
  3012.     
  3013. /* Map a color index to a r-g-b color. */
  3014. private int
  3015. gdev_pcl_map_color_rgb(gx_device *pdev, gx_color_index color,
  3016.                 gx_color_value prgb[3])
  3017. {
  3018.   /* For the moment, we simply ignore any black correction */
  3019.   switch (pdev->color_info.depth) {
  3020.   case 1:
  3021.     prgb[0] = prgb[1] = prgb[2] = -((gx_color_value)color ^ 1);
  3022.     break;
  3023.   case 8:
  3024.       if (pdev->color_info.num_components >= 3)
  3025.     { gx_color_value c = (gx_color_value)color ^ 7;
  3026.       prgb[0] = -(c & 1);
  3027.       prgb[1] = -((c >> 1) & 1);
  3028.       prgb[2] = -(c >> 2);
  3029.     }
  3030.       else
  3031.     { gx_color_value value = (gx_color_value)color ^ 0xff;
  3032.       prgb[0] = prgb[1] = prgb[2] = (value << 8) + value;
  3033.     }
  3034.     break;
  3035.   case 16:
  3036.     { gx_color_value c = (gx_color_value)color ^ 0xffff;
  3037.       ushort value = c >> 11;
  3038.       prgb[0] = ((value << 11) + (value << 6) + (value << 1) +
  3039.          (value >> 4)) >> (16 - gx_color_value_bits);
  3040.       value = (c >> 6) & 0x3f;
  3041.       prgb[1] = ((value << 10) + (value << 4) + (value >> 2))
  3042.     >> (16 - gx_color_value_bits);
  3043.       value = c & 0x1f;
  3044.       prgb[2] = ((value << 11) + (value << 6) + (value << 1) +
  3045.          (value >> 4)) >> (16 - gx_color_value_bits);
  3046.     }
  3047.     break;
  3048.   case 24:
  3049.     { gx_color_value c = (gx_color_value)color ^ 0xffffff;
  3050.       prgb[0] = gx_color_value_from_byte(c >> 16);
  3051.       prgb[1] = gx_color_value_from_byte((c >> 8) & 0xff);
  3052.       prgb[2] = gx_color_value_from_byte(c & 0xff);
  3053.     }
  3054.     break;
  3055.   case 32:
  3056. #define  gx_maxcol gx_color_value_from_byte(gx_color_value_to_byte(gx_max_color_value))
  3057.     { gx_color_value w = gx_maxcol - gx_color_value_from_byte(color >> 24);
  3058.       prgb[0] = w - gx_color_value_from_byte((color >> 16) & 0xff);
  3059.       prgb[1] = w - gx_color_value_from_byte((color >> 8) & 0xff);
  3060.       prgb[2] = w - gx_color_value_from_byte(color & 0xff);
  3061.     }
  3062.     break;
  3063.   }
  3064.   return 0;
  3065. }
  3066.  
  3067. /*
  3068.  * Convert and expand scanlines:
  3069.  *
  3070.  * For devices with 3 components:
  3071.  *
  3072.  *       (a)    16 -> 24 bit   (1-stage)
  3073.  *       (b)    16 -> 32 bit   (2-stage)
  3074.  *   or  (c)    24 -> 32 bit   (1-stage)
  3075.  *
  3076.  * For devices with 4 components:
  3077.  *
  3078.  *       (a)    16 -> 32 bit   (1-stage)
  3079.  *       (b)     8 -> 32 bit   (2-stage)
  3080.  *  or   (c)    24 -> 32 bit   (1-stage)
  3081.  *
  3082.  */
  3083.  
  3084. private void
  3085. cdj_expand_line(word *line, int linesize, short cmyk, int bpp, int ebpp)
  3086. {
  3087.   int endline = linesize;
  3088.   byte *start = (byte *)line;
  3089.   register byte *in, *out;
  3090.  
  3091.   if (cmyk > 0) {
  3092.       if (bpp == 8) {
  3093.       in = start + endline;
  3094.       out = start + (endline *=  2);
  3095.  
  3096.       while (in > start) {
  3097.           register byte b0;
  3098.           register byte bs0, bs1, bs2, bs3;
  3099.  
  3100.           b0 = *--in;
  3101.  
  3102.           bs0 = b0 & 0x03;
  3103.           bs1 = (b0 >> 2) & 0x03;
  3104.           bs2 = (b0 >> 4) & 0x03;
  3105.           bs3 = (b0 >> 6) & 0x03;
  3106.           
  3107.           *--out = (bs0 << 2) + bs0 + (bs1 << 6) + (bs1 << 4);
  3108.           *--out = (bs2 << 2) + bs2 + (bs3 << 6) + (bs3 << 4);
  3109.       }
  3110.       }
  3111.  
  3112.       if (bpp == 24) {
  3113.       endline = (endline + 2) / 3;
  3114.       
  3115.       in = start + endline * 3;
  3116.       out = start + endline * 4;
  3117.  
  3118.       while (in > start) {
  3119.           register byte b0, b1, b2;
  3120.  
  3121.           b0 = *--in;
  3122.           b1 = *--in;
  3123.           b2 = *--in;
  3124.  
  3125.           *--out = (b0 << 2) + ((b0 >> 4) & 0x03);
  3126.           *--out = ((b1 & 0x0f) << 4) + ((b0 >> 6) << 2)
  3127.               + ((b1 >> 2) & 0x03);
  3128.           *--out = ((b2 & 0x03) << 6) + ((b1 >> 4) << 2) + (b2 & 0x03);
  3129.           *--out = (b2 & 0xfc) + ((b2 >> 6) & 0x03);
  3130.       }
  3131.       } else if (ebpp == 32) {
  3132.       endline = (endline + 1) / 2;
  3133.  
  3134.       in = start + endline * 2;
  3135.       out = start + (endline *= 4);
  3136.  
  3137.       while (in > start) {
  3138.           register byte b0, b1;
  3139.  
  3140.           b0 = *--in;
  3141.           b1 = *--in;
  3142.  
  3143.           *--out = (b0 << 4) + ((b0 >> 4) & 0x07);
  3144.           *--out = (b0 & 0xf0) + ((b0 >> 4) & 0xf);
  3145.           *--out = (b1 << 4) + ((b1 >> 4) & 0x0f);
  3146.           *--out = (b1 & 0xf0) + ((b1 >> 4) & 0xf);
  3147.       }
  3148.       }
  3149.   } else /* cmyk > 0 */ {
  3150.       if (bpp == 16)              /* 16 to 24 (cmy) if required */
  3151.       { register byte b0, b1;
  3152.         endline = ((endline + 1) / 2);
  3153.         in = start + endline * 2;
  3154.         out = start + (endline *= 3);
  3155.       
  3156.         while (in > start)
  3157.         { b0 = *--in;
  3158.           b1 = *--in;
  3159.           *--out = (b0 << 3) + ((b0 >> 2) & 0x7);
  3160.           *--out = (b1 << 5) + ((b0 >> 3)  & 0x1c) + ((b1 >> 1) & 0x3);
  3161.           *--out = (b1 & 0xf8) + (b1 >> 5);
  3162.           }
  3163.     }
  3164.  
  3165.       if (ebpp == 32)             /* 24/32 (cmy) to 32 (cmyk) if required */
  3166.       { register byte c, m, y;
  3167.         endline = ((endline + 2) / 3);
  3168.         in = start + endline * 3;
  3169.         out = start + endline * 4;
  3170.  
  3171.         while (in > start)
  3172.         { 
  3173.           y = *--in;
  3174.           m = *--in;
  3175.           c = *--in;
  3176.  
  3177.           if (c == y && c == m) {
  3178.               *--out = 0, *--out = 0, *--out = 0;
  3179.               *--out = c;
  3180.           } else {
  3181.               *--out = y, *--out = m, *--out = c;
  3182.               *--out = 0;
  3183.           }
  3184.           }
  3185.     }
  3186.   }
  3187. }
  3188.  
  3189. private int near
  3190. cdj_put_param_int(gs_param_list *plist, gs_param_name pname, int *pvalue,
  3191.   int minval, int maxval, int ecode)
  3192. {    int code, value;
  3193.     switch ( code = param_read_int(plist, pname, &value) )
  3194.     {
  3195.     default:
  3196.         return code;
  3197.     case 1:
  3198.         return ecode;
  3199.     case 0:
  3200.         if ( value < minval || value > maxval )
  3201.            param_signal_error(plist, pname, gs_error_rangecheck);
  3202.         *pvalue = value;
  3203.         return (ecode < 0 ? ecode : 1);
  3204.     }
  3205. }    
  3206.  
  3207. private int
  3208. cdj_set_bpp(gx_device *pdev, int bpp, int ccomps)
  3209. { gx_device_color_info *ci = &pdev->color_info;
  3210.  
  3211.   if (ccomps && bpp == 0) {
  3212.       if (cprn_device->cmyk) {
  3213.       switch (ccomps) {
  3214.           default:
  3215.               return gs_error_rangecheck;
  3216.           /*NOTREACHED*/
  3217.           break;
  3218.  
  3219.           case 1:
  3220.               bpp = 1;
  3221.           break;
  3222.  
  3223.           case 3:
  3224.           bpp = 24;
  3225.           break;
  3226.  
  3227.           case 4:
  3228.           switch (ci->depth) {
  3229.               case 8:
  3230.               case 16:
  3231.               case 24:
  3232.               case 32:
  3233.                   break;
  3234.  
  3235.               default:
  3236.               bpp = cprn_device->default_depth;
  3237.               break;
  3238.           }
  3239.           break;
  3240.       }
  3241.       }
  3242.   }
  3243.  
  3244.   if (bpp == 0) {
  3245.       bpp = ci->depth;        /* Use the current setting. */
  3246.   }
  3247.  
  3248.   if (cprn_device->cmyk < 0) {
  3249.  
  3250.       /* Reset procedures because we may have been in another mode. */
  3251.  
  3252.       dev_proc(pdev, map_cmyk_color) = gdev_cmyk_map_cmyk_color;
  3253.       dev_proc(pdev, map_rgb_color) = NULL;
  3254.       dev_proc(pdev, map_color_rgb) = gdev_cmyk_map_color_rgb;
  3255.  
  3256.       if (pdev->is_open) gs_closedevice(pdev);
  3257.   }
  3258.  
  3259.   /* Check for valid bpp values */
  3260.  
  3261.   switch ( bpp )
  3262.     {
  3263.     case 16:
  3264.     case 32:
  3265.     if (cprn_device->cmyk && ccomps && ccomps != 4) goto bppe;
  3266.     break;
  3267.  
  3268.     case 24:
  3269.        if (!cprn_device->cmyk || ccomps == 0 || ccomps == 4) {
  3270.        break;
  3271.        } else if (ccomps == 1) {
  3272.        goto bppe;
  3273.        } else {
  3274.  
  3275.        /* 3 components 24 bpp printing for CMYK device. */
  3276.  
  3277.        cprn_device->cmyk = -1;
  3278.        }
  3279.        break;
  3280.  
  3281.     case 8:
  3282.     if (cprn_device->cmyk) {
  3283.         if (ccomps) {
  3284.         if (ccomps == 3) {
  3285.             cprn_device->cmyk = -1;
  3286.             bpp = 3;
  3287.         } else if (ccomps != 1 && ccomps != 4) {
  3288.             goto bppe;
  3289.         }
  3290.         }
  3291.         if (ccomps != 1) break;
  3292.     } else {
  3293.         break;
  3294.     }
  3295.  
  3296.     case 1:
  3297.        if (ccomps != 1) goto bppe;
  3298.  
  3299.        if (cprn_device->cmyk && bpp != pdev->color_info.depth) {
  3300.        dev_proc(pdev, map_cmyk_color) = NULL;
  3301.        dev_proc(pdev, map_rgb_color) = gdev_cmyk_map_rgb_color;
  3302.  
  3303.        if (pdev->is_open) {
  3304.            gs_closedevice(pdev);
  3305.        }
  3306.        }
  3307.        break;
  3308.  
  3309.     case 3:
  3310.     if (!cprn_device->cmyk) {
  3311.         break;
  3312.     }
  3313.  
  3314.     default:
  3315. bppe:  return gs_error_rangecheck;
  3316.     }
  3317.  
  3318.  
  3319.     if (cprn_device->cmyk == -1) {
  3320.     dev_proc(pdev, map_cmyk_color) = NULL;
  3321.     dev_proc(pdev, map_rgb_color) = gdev_pcl_map_rgb_color;
  3322.     dev_proc(pdev, map_color_rgb) = gdev_pcl_map_color_rgb;
  3323.  
  3324.     if (pdev->is_open) {
  3325.         gs_closedevice(pdev);
  3326.     }
  3327.     }
  3328.  
  3329.   switch (ccomps) {
  3330.       case 0:
  3331.           break;
  3332.  
  3333.       case 1:
  3334.       if (bpp != 1 && bpp != 8) goto cce;
  3335.       break;
  3336.  
  3337.       case 4:
  3338.       if (cprn_device->cmyk) {
  3339.           if (bpp >= 8) break;
  3340.       }
  3341.  
  3342.       case 3:
  3343.       if (bpp == 1 || bpp == 3 || bpp == 8 || bpp == 16
  3344.           || bpp == 24 || bpp == 32) {
  3345.           break;
  3346.       }
  3347.  
  3348. cce:  default: return gs_error_rangecheck;
  3349.   }
  3350.  
  3351.   if (cprn_device->cmyk) {
  3352.       if (cprn_device->cmyk > 0) {
  3353.       ci->num_components = ccomps ? ccomps : (bpp < 8 ? 1 : 4);
  3354.       } else {
  3355.       ci->num_components = ccomps ? ccomps : (bpp < 8 ? 1 : 3);
  3356.       }
  3357.       if (bpp != 1 && ci->num_components == 1) { /* We do dithered grays. */
  3358.       bpp = bpp < 8 ? 8 : bpp;
  3359.       }
  3360.  
  3361.       ci->max_color = (1 << (bpp >> 2)) - 1;
  3362.       ci->max_gray = (bpp >= 8 ? 255 : 1);
  3363.  
  3364.       if (ci->num_components == 1) {
  3365.       ci->dither_grays = (bpp >= 8 ? 5 : 2);
  3366.       ci->dither_colors = (bpp >= 8 ? 5 : bpp > 1 ? 2 : 0);
  3367.       } else {
  3368.       ci->dither_grays = (bpp > 8 ? 5 : 2);
  3369.       ci->dither_colors = (bpp > 8 ? 5 : bpp > 1 ? 2 : 0);
  3370.       }
  3371.   } else {
  3372.       ci->num_components = (bpp == 1 || bpp == 8 ? 1 : 3);
  3373.       ci->max_color = (bpp >= 8 ? 255 : bpp > 1 ? 1 : 0);
  3374.       ci->max_gray = (bpp >= 8 ? 255 : 1);
  3375.       ci->dither_grays = (bpp >= 8 ? 5 : 2);
  3376.       ci->dither_colors = (bpp >= 8 ? 5 : bpp > 1 ? 2 : 0);
  3377.   }
  3378.  
  3379.   ci->depth = ((bpp > 1) && (bpp < 8) ? 8 : bpp);
  3380.  
  3381.   return 0;
  3382. }
  3383.  
  3384. /* new_bpp == save_bpp or new_bpp == 0 means don't change bpp.
  3385.    ccomps == 0 means don't change number of color comps.
  3386.    If new_bpp != 0, it must be the value of the BitsPerPixel element of
  3387.      the plist; real_bpp may differ from new_bpp.
  3388. */
  3389. private int
  3390. cdj_put_param_bpp(gx_device *pdev, gs_param_list *plist, int new_bpp,
  3391.   int real_bpp, int ccomps)
  3392. {
  3393.     if (new_bpp == 0 && ccomps == 0)
  3394.       return gdev_prn_put_params(pdev, plist);
  3395.     else
  3396.       {
  3397.         gx_device_color_info save_info;
  3398.         int save_bpp;
  3399.         int code;
  3400.  
  3401.         save_info = pdev->color_info;
  3402.         save_bpp = save_info.depth;
  3403. #define save_ccomps save_info.num_components
  3404.         if ( save_bpp == 8 && save_ccomps == 3 && !cprn_device->cmyk)
  3405.           save_bpp = 3;
  3406.         code = cdj_set_bpp(pdev, real_bpp, ccomps);
  3407.         if ( code < 0 ) {
  3408.           param_signal_error(plist, "BitsPerPixel", code);
  3409.           param_signal_error(plist, "ProcessColorModel", code);
  3410.           return code;
  3411.             }
  3412.         pdev->color_info.depth = new_bpp;  /* cdj_set_bpp maps 3/6 to 8 */
  3413.         code = gdev_prn_put_params(pdev, plist);
  3414.         if ( code < 0 )
  3415.           {    cdj_set_bpp(pdev, save_bpp, save_ccomps);
  3416.             return code;
  3417.           }
  3418.         cdj_set_bpp(pdev, real_bpp, ccomps);    /* reset depth if needed */
  3419.         if ((cdj->color_info.depth != save_bpp ||
  3420.              (ccomps != 0 && ccomps != save_ccomps))
  3421.             && pdev->is_open )
  3422.           return gs_closedevice(pdev);
  3423.         return 0;
  3424. #undef save_ccomps
  3425.       }
  3426. }
  3427.  
  3428. /* This returns either the number of pixels in a scan line, or the number
  3429.  * of bytes required to store the line, both clipped to the page margins */
  3430. private uint
  3431. gdev_prn_rasterwidth(const gx_device_printer *pdev, int pixelcount)
  3432. {
  3433.   ulong raster_width =
  3434.     pdev->width - pdev->x_pixels_per_inch * (dev_l_margin(pdev) + dev_r_margin(pdev));
  3435.   return (pixelcount ?
  3436.           (uint)raster_width :
  3437.           (uint)((raster_width * pdev->color_info.depth + 7) >> 3));
  3438. }
  3439.  
  3440. /* Functions for manipulation params strings */
  3441.  
  3442. private const byte*
  3443. paramValueToString(const stringParamDescription* params, int value)
  3444. {
  3445.  
  3446.     for (; params->p_name; ++params) {
  3447.     if (params->p_value == value) {
  3448.         return (const byte *)params->p_name;
  3449.     }
  3450.     }
  3451.  
  3452.     return (const byte*) 0;
  3453. }
  3454.  
  3455. private int
  3456. paramStringValue(const stringParamDescription* params,
  3457.     const byte* name, int namelen, int* value)
  3458. {
  3459.  
  3460.     for (; params->p_name; ++params) {
  3461.     if (strncmp(params->p_name, (char *)name, namelen) == 0 &&
  3462.         params->p_name[namelen] == 0) {
  3463.         *value = params->p_value;
  3464.         return 1;
  3465.     }
  3466.     }
  3467.  
  3468.     return 0;
  3469. }
  3470.  
  3471. private int
  3472. put_param_string(gs_param_list* plist,
  3473.     const byte* pname, gs_param_string* pstring,
  3474.     const stringParamDescription* params, int *pvalue, int code)
  3475. {
  3476.  
  3477.     int ncode;
  3478.  
  3479.     if ((ncode = param_read_string(plist, (char *)pname, pstring)) < 0) {
  3480.     param_signal_error(plist, (char *)pname, code = ncode);
  3481.     } else if (ncode == 1) {
  3482.     pstring->data = 0, pstring->size = 0;
  3483.     } else {
  3484.     int value = 0;
  3485.  
  3486.     if (paramStringValue(params, pstring->data, pstring->size,
  3487.             &value) == 0) {
  3488.         param_signal_error(plist, (char *)pname, code = gs_error_rangecheck);
  3489.     } else {
  3490.         *pvalue = value;
  3491.     }
  3492.     }
  3493.  
  3494.     return code;
  3495. }
  3496.  
  3497. private int
  3498. get_param_string(gs_param_list* plist,
  3499.     const byte* pname, gs_param_string* pstring,
  3500.     const stringParamDescription* params, int pvalue, bool persist, int code)
  3501. {
  3502.  
  3503.     int ncode;
  3504.  
  3505.     pstring->data = paramValueToString(params, pvalue);
  3506.  
  3507.     if (pstring->data == (byte*) 0) {
  3508.     param_signal_error(plist, (char *)pname, ncode = gs_error_unknownerror);
  3509.     } else {
  3510.         pstring->size = strlen((char *)pstring->data);
  3511.     pstring->persistent = persist;
  3512.     }
  3513.  
  3514.     if ((ncode = param_write_string(plist, (char *)pname, pstring)) < 0) {
  3515.     code = ncode;
  3516.     }
  3517.  
  3518.     return code;
  3519. }
  3520.  
  3521. /*
  3522.  * This taken from gsdparam.c. I hope it will be useable directly some day.
  3523.  *
  3524.  */
  3525.  
  3526. private int
  3527. cdj_param_check_bytes(gs_param_list *plist, gs_param_name pname, const byte *str,
  3528.   uint size, bool defined)
  3529. {       int code;
  3530.         gs_param_string new_value;
  3531.         switch ( code = param_read_string(plist, pname, &new_value) )
  3532.           {
  3533.           case 0:
  3534.                 if ( defined && new_value.size == size &&
  3535.                      !memcmp((const char *)str, (const char *)new_value.data,
  3536.                              size)
  3537.                    )
  3538.                   break;
  3539.                 code = gs_note_error(gs_error_rangecheck);
  3540.                 goto e;
  3541.           default:
  3542.                 if ( param_read_null(plist, pname) == 0 )
  3543.                   return 1;
  3544. e:              param_signal_error(plist, pname, code);
  3545.           case 1:
  3546.                 ;
  3547.           }
  3548.         return code;
  3549. }
  3550.  
  3551. /* This is original code. */
  3552.  
  3553. private int
  3554. cdj_param_check_float(gs_param_list *plist, gs_param_name pname, floatp fval,
  3555.   bool defined)
  3556. {       int code;
  3557.         float new_value;
  3558.         switch ( code = param_read_float(plist, pname, &new_value) )
  3559.           {
  3560.           case 0:
  3561.                 if ( defined && new_value == fval)
  3562.                   break;
  3563.                 code = gs_note_error(gs_error_rangecheck);
  3564.                 goto e;
  3565.           default:
  3566.                 if ( param_read_null(plist, pname) == 0 )
  3567.                   return 1;
  3568. e:              param_signal_error(plist, pname, code);
  3569.           case 1:
  3570.                 ;
  3571.           }
  3572.         return code;
  3573. }
  3574.  
  3575. /* The following dithering algorithm has been kindly given to me (YA) by
  3576.  * Klaus-Gunther Hess, I just adapted it for use with the code here. */
  3577.  
  3578. /*
  3579.  
  3580. (From KGH:)
  3581.  
  3582. Just about the features of the code:
  3583.  
  3584.     - Stored Color-Values are BYTES in the order C-M-Y-K.
  3585.       (Indices need to change with gdevcdj.c)
  3586.  
  3587.     - There are individual THRESHOLDs and SPOTSIZEs for
  3588.       the color-components. The following relation should
  3589.       be maintained:
  3590.                  SPOTSIZE = 2 * THRESHOLD + 1
  3591.       (The internal calculation is dedicated for limiting
  3592.        ink-density at the 720x720DpI-Resolution of the 
  3593.        Epson-Printers, without loss of dynamic color-range)
  3594.  
  3595.     - In addition to that there are EMIN & EMAX-Values 
  3596.       for the components. The Values are computed from
  3597.       the dithering-algorithm and can be replaced by
  3598.       constants, if neither the implementation nor 
  3599.       THRESHOLD and SPOTSIZE can change.
  3600.  
  3601.     - The algorithm is tuned for speed. (K-only, if gray-
  3602.       levels are detected, with EMIN/EMAX-clipping of
  3603.       stored CMY-Errors. [Notice: cerr, merr, yerr are
  3604.       *not* reset to zero! Clearing them would cause 
  3605.       regular patterns & "Halos" to appear!])
  3606.  
  3607. */
  3608.  
  3609. /*
  3610.  * Macros, that represent the undisturbed dithering-algorithm
  3611.  *
  3612.  * FSerror:   compute the desired Value
  3613.  * FSdecide:  decision based on the value computed by FSerror
  3614.  * FSdiffuse: distribute remaining error among pixels
  3615.  */
  3616.  
  3617. #define FSerror(Val,Erow,Ecol) (Val + Erow + ((7 * Ecol)>>4))
  3618.  
  3619. #define FSdecide(Error,Threshold,Spotsize,Pixel,Bit) \
  3620.          if(Error > Threshold) {\
  3621.             Pixel |= Bit;\
  3622.             Error -= Spotsize;\
  3623.          }
  3624.  
  3625. #define FSdiffuse(Error,Erow,Ecol,Eprev)\
  3626.          Eprev  += (3 * Error + 8)>>4;\
  3627.          Erow    = (5 * Error + Ecol + 8)>>4;\
  3628.          Ecol    = Error;
  3629.  
  3630. /*
  3631.  * some aliases for values from the device-structure
  3632.  */
  3633. #define DIRECTION    direction[0]
  3634. #define CMYK_THRESHOLD(I) threshold[I]
  3635. #define SPOTSIZE(I)  spotsize[I]
  3636. #define EMIN(I)      emin[I]
  3637. #define EMAX(I)      emax[I]
  3638. #define NPIXEL       (plane_size * 8)
  3639.  
  3640. #define IDX_C         1
  3641. #define IDX_M         2
  3642. #define IDX_Y         3
  3643. #define IDX_K         0
  3644.  
  3645. #define ODX_C         2
  3646. #define ODX_M         1
  3647. #define ODX_Y         0
  3648. #define ODX_K         3
  3649.  
  3650. private int
  3651. bjc_fscmyk(byte** inplanes, byte* outplanes[4][4], int** errplanes,
  3652.     int plane_size, int scan) {
  3653.  
  3654.     byte* err = (byte*) errplanes[0];
  3655.  
  3656. /* =========================================================== */
  3657.    if(scan < 0) { /* scan <   0 -> initialize private buffer */
  3658. /* =========================================================== */
  3659.  
  3660.       int p,i,v;
  3661.       int *direction,*threshold,*spotsize,*emin,*emax;
  3662.       int *errv,*errc;
  3663. /*
  3664.  * allocate the error-buffer
  3665.  */
  3666.       /*KGHorig
  3667.       i = 4 * (5 + 1 + 1 + sd->stc.prt_pixels + 1) * sizeof(errv[0]);
  3668.       if((sd->stc.err_size < i) || (NULL == sd->stc.err)) {
  3669.          if(NULL != sd->stc.err) 
  3670.             gs_free(sd->stc.err,sd->stc.err_size,1,"stcm/err");
  3671.          sd->stc.err_size = i;
  3672.          sd->stc.err = gs_malloc(sd->stc.err_size,1,"stcm/err");
  3673.          if(sd->stc.err == NULL) return_error(gs_error_VMerror);
  3674.       }
  3675.       */
  3676.  
  3677.       direction = (int *) err;
  3678.       threshold = direction + 4;
  3679.       spotsize  = threshold + 4;
  3680.       emin      = spotsize  + 4;
  3681.       emax      = emin      + 4;
  3682.       errc      = emax      + 4;
  3683.       errv      = errc      + 2*4;
  3684. /*
  3685.  * compute initial values
  3686.  */
  3687.       DIRECTION = -1;
  3688.       for(i = 0; i < 4; ++i) {
  3689.          int j;
  3690.          float maxv = 1.0;
  3691.      /*KGHorig
  3692.          if((sd->stc.xfer[i].size < 1) || (sd->stc.xfer[i].data == NULL)) {
  3693.             maxv = 1.0;
  3694.          } else {
  3695.             maxv = 1.0/255.0;
  3696.             for(j = 0; j < sd->stc.xfer[i].size; ++j)
  3697.                if(maxv < sd->stc.xfer[i].data[j])
  3698.              maxv = sd->stc.xfer[i].data[j];
  3699.          }
  3700.      */
  3701.          CMYK_THRESHOLD(i) = 127.0 / maxv + 0.5;
  3702.          SPOTSIZE(i)  = ((int) CMYK_THRESHOLD(i)<<1)+1;
  3703.          j = CMYK_THRESHOLD(i); /* Maximum Error-Value */
  3704.          errc[3] = 0;
  3705.          FSdiffuse(CMYK_THRESHOLD(i),errv[0],errc[0],errv[-4]);
  3706.          FSdiffuse(CMYK_THRESHOLD(i),errv[0],errc[0],errv[-4]);
  3707.          EMAX(i) = errv[0];
  3708.          errc[0] = 0;
  3709.          FSdiffuse((-CMYK_THRESHOLD(i)),errv[0],errc[0],errv[-4]);
  3710.          FSdiffuse((-CMYK_THRESHOLD(i)),errv[0],errc[0],errv[-4]);
  3711.          EMIN(i) = errv[0];
  3712.       }
  3713.  
  3714. #ifdef CDJ_DEBUG_FS
  3715.       for(i = 0; i < 4; ++i) fprintf(stderr,
  3716.          "CMYK_THRESHOLD(%d)=%5d, spotsize(%d)=%5d, emin(%d)=%5d, emax(%d)=%5d\n",
  3717.          i,CMYK_THRESHOLD(i),i,SPOTSIZE(i),i,EMIN(i),i,EMAX(i));
  3718. #endif
  3719.  
  3720.       for(i = 0; i < 4; ++i) errc[i] = 0;
  3721.  
  3722.       for(p = 0; p < NPIXEL; ++p) {
  3723.          for(i = 0; i < 4; ++i) {
  3724.         /*KHGOrig
  3725.             if(sd->stc.flags & STCDFLAG0) v = 0;
  3726.         */
  3727.         if (0) v = 0;    /* Must provide a default for that. */
  3728.             else                      v = (rand() % SPOTSIZE(i)) - CMYK_THRESHOLD(i);
  3729.             FSdiffuse(v,errv[i],errc[i],errv[i-4]);
  3730.          }
  3731.          errv += i;
  3732.       }
  3733.  
  3734. /* =========================================================== */
  3735.    } else {        /* scan >=  0 -> scanline-processing       */
  3736. /* =========================================================== */
  3737.  
  3738.       int w,p,dir,thedir;
  3739.       byte *out[4],pixel[4],bit;
  3740.       /*KGHorig
  3741.       int  *width     = outplanes[scan];
  3742.       */
  3743.       int  *direction = (int *) err;
  3744.       int  *threshold = direction + 4;
  3745.       int  *spotsize  = threshold + 4;
  3746.       int  *emin      = spotsize  + 4;
  3747.       int  *emax      = emin      + 4;
  3748.       int  *errc      = emax      + 4;
  3749.       int  *errv      = errc      + 2*4;
  3750.       int   kerr,cerr,merr,yerr;
  3751.  
  3752.       byte* in;
  3753.  
  3754.       /*KGHorig
  3755.       if(sd->stc.flags & STCDFLAG1) {
  3756.       */
  3757.       if (0) {            /* Eventually will provide a flag for this. */
  3758.          cerr = merr = yerr = kerr = 0;
  3759.       } else {
  3760.          cerr = errc[0];
  3761.          merr = errc[1];
  3762.          yerr = errc[2];
  3763.          kerr = errc[3];
  3764.       }
  3765.  
  3766.       out[0]   = outplanes[scan + 2][ODX_C];
  3767.       out[1]   = outplanes[scan + 2][ODX_M];
  3768.       out[2]   = outplanes[scan + 2][ODX_Y];
  3769.       out[3]   = outplanes[scan + 2][ODX_K];
  3770.       pixel[0] = pixel[1] = pixel[2] = pixel[3] = 0;
  3771.  
  3772.       if(DIRECTION < 0) { /* scan == 0, run backward */
  3773.          w      = NPIXEL;
  3774.      in     = inplanes[2] + 4 * (NPIXEL - 1);
  3775.          errv  += (w-1)<<2;
  3776.          dir    = -4;
  3777.      /*KGHorig
  3778.          if(w > 8) for(p = 0; p < 4; ++p) out[p] += (w-1)>>3;
  3779.      */
  3780.      thedir = -1;
  3781.      for (p = 0; p < 4; ++p) {
  3782.          out[p] += plane_size - 1;
  3783.      }
  3784.       } else {      /* run forward */
  3785.          w      = 1;
  3786.      in      = inplanes[3] - 4 * NPIXEL;
  3787.          dir    = 4;
  3788.      thedir = 1;
  3789.      for (p = 0; p < 4; ++p) {
  3790.          out[p] -= plane_size;
  3791.      }
  3792.       }             /* run backward/forward */
  3793.  
  3794.       /*KGHorig
  3795.       if(0 == (sd->stc.flags & STCDFLAG1)) DIRECTION = -DIRECTION;
  3796.       */
  3797.       if (1) DIRECTION = -DIRECTION; /* Scan in other direction. */
  3798.  
  3799.       bit = 0x80>>((w-1) & 7);
  3800.       w   = (w+7)>>3;
  3801.  
  3802.       for(p = NPIXEL; p; --p) { /* loop over pixels */
  3803.  
  3804.          int cmy = in[IDX_C] | in[IDX_M] | in[IDX_Y];
  3805.          int kv  = FSerror(in[IDX_K],errv[3],kerr);
  3806.          int cv;
  3807.  
  3808.          FSdecide(kv,CMYK_THRESHOLD(3),SPOTSIZE(3),pixel[3],bit);
  3809.  
  3810.          if(cmy) {
  3811.  
  3812.             if(pixel[3] & bit) { /* black known to fire */
  3813.  
  3814.                FSdiffuse(kv,errv[3],kerr,errv[3-dir]);
  3815.  
  3816.                cv  = FSerror(in[IDX_C],errv[0],cerr);
  3817.                cv -= SPOTSIZE(0); 
  3818.                if ((cv+CMYK_THRESHOLD(0)) < 0) cv = -CMYK_THRESHOLD(0);
  3819.                FSdiffuse(cv,errv[0],cerr,errv[0-dir]);
  3820.  
  3821.                cv  = FSerror(in[IDX_M],errv[1],merr);
  3822.                cv -= SPOTSIZE(1);
  3823.                if ((cv+CMYK_THRESHOLD(1)) < 0) cv = -CMYK_THRESHOLD(1);
  3824.  
  3825.                FSdiffuse(cv,errv[1],merr,errv[1-dir]);
  3826.  
  3827.                cv  = FSerror(in[IDX_Y],errv[2],yerr);
  3828.                cv -= SPOTSIZE(2);
  3829.                if ((cv+CMYK_THRESHOLD(2)) < 0) cv = -CMYK_THRESHOLD(2);
  3830.                FSdiffuse(cv,errv[2],yerr,errv[2-dir]);
  3831.  
  3832.             } else {
  3833.  
  3834.                cv  = FSerror(in[IDX_C],errv[0],cerr);
  3835.                FSdecide(cv,CMYK_THRESHOLD(0),SPOTSIZE(0),pixel[0],bit);
  3836.                FSdiffuse(cv,errv[0],cerr,errv[0-dir]);
  3837.  
  3838.                cv  = FSerror(in[IDX_M],errv[1],merr);
  3839.                FSdecide(cv,CMYK_THRESHOLD(1),SPOTSIZE(1),pixel[1],bit);
  3840.                FSdiffuse(cv,errv[1],merr,errv[1-dir]);
  3841.  
  3842.                cv  = FSerror(in[IDX_Y],errv[2],yerr);
  3843.                FSdecide(cv,CMYK_THRESHOLD(2),SPOTSIZE(2),pixel[2],bit);
  3844.                FSdiffuse(cv,errv[2],yerr,errv[2-dir]);
  3845.  
  3846.                if(pixel[0] & pixel[1] & pixel[2] & bit) {
  3847.                   pixel[0] &= ~bit;
  3848.                   pixel[1] &= ~bit;
  3849.                   pixel[2] &= ~bit;
  3850.                   pixel[3] |=  bit;
  3851.                   kv -= SPOTSIZE(3);
  3852.                   if ((kv+CMYK_THRESHOLD(3)) < 0) kv = -CMYK_THRESHOLD(0);  
  3853.                   FSdiffuse(kv,errv[3],kerr,errv[3-dir]);
  3854.                }
  3855.        }
  3856.  
  3857.          } else {
  3858.  
  3859.             FSdiffuse(kv,errv[3],kerr,errv[3-dir]);
  3860.  
  3861.             if(     errv[0] > EMAX(0)) errv[0] = EMAX(0);
  3862.             else if(errv[0] < EMIN(0)) errv[0] = EMIN(0);
  3863.  
  3864.             if(     errv[1] > EMAX(1)) errv[1] = EMAX(1);
  3865.             else if(errv[1] < EMIN(1)) errv[1] = EMIN(1);
  3866.  
  3867.             if(     errv[2] > EMAX(2)) errv[2] = EMAX(2);
  3868.             else if(errv[2] < EMIN(2)) errv[2] = EMIN(2);
  3869.  
  3870.          }
  3871.  
  3872. /*
  3873.  * Adjust indices
  3874.  */
  3875.          bit   = dir > 0 ? (bit>>1) : (bit<<1);
  3876.          if(bit == 0) {
  3877.         /*KGHorig
  3878.             if(((*out[0]  = pixel[0]) != 0) && (width[0] < w)) width[0] = w;
  3879.             if(((*out[1]  = pixel[1]) != 0) && (width[1] < w)) width[1] = w;
  3880.             if(((*out[2]  = pixel[2]) != 0) && (width[2] < w)) width[2] = w;
  3881.             if(((*out[3]  = pixel[3]) != 0) && (width[3] < w)) width[3] = w;
  3882.         */
  3883.         *out[0] = pixel[0];
  3884.         *out[1] = pixel[1];
  3885.         *out[2] = pixel[2];
  3886.         *out[3] = pixel[3];
  3887.             out[0] += thedir; out[1] += thedir;
  3888.             out[2] += thedir; out[3] += thedir;
  3889.             pixel[0] = pixel[1] = pixel[2] = pixel[3] = 0;
  3890.  
  3891.             if(dir > 0) bit = 0x80;
  3892.             else        bit = 0x01;
  3893.             w    += dir>>2;
  3894.          }
  3895.  
  3896.      in += dir;
  3897.          errv += dir;
  3898.       }                                         /* loop over pixels */
  3899.  
  3900.       /*KGHorig
  3901.       if(0 == (sd->stc.flags & STCDFLAG1)) {
  3902.       */
  3903.       if (1) {
  3904.          cerr = errc[0] = cerr;
  3905.          merr = errc[1] = merr;
  3906.          yerr = errc[2] = yerr;
  3907.          kerr = errc[3] = kerr;
  3908.       }
  3909.  
  3910. /* =========================================================== */
  3911.    }                  /* initialization or scanline-Processing */
  3912. /* =========================================================== */
  3913.  
  3914.    return 0;
  3915. }
  3916.  
  3917.